跨域一词从字面意思看,就是跨域名嘛,但实际上跨域的范围绝对不止那么狭隘。具体概念如下:只要协议、域名、端口有任何一个不同,都被当作是不同的域。
URL 说明 是否允许通信
http://www.a.com/a.js
http://www.a.com/b.js 同一域名下 允许
http://www.a.com/lab/a.js
http://www.a.com/script/b.js 同一域名下不同文件夹 允许
http://www.a.com:8000/a.js
http://www.a.com/b.js 同一域名,不同端口 不允许
http://www.a.com/a.js
https://www.a.com/b.js 同一域名,不同协议 不允许
http://www.a.com/a.js
http://70.32.92.74/b.js 域名和域名对应ip 不允许
http://www.a.com/a.js
http://script.a.com/b.js 主域相同,子域不同 不允许(cookie这种情况下也不允许访问)
http://www.a.com/a.js
http://a.com/b.js 同一域名,不同二级域名(同上) 不允许(cookie这种情况下也不允许访问)
http://www.cnblogs.com/a.js
http://www.a.com/b.js 不同域名 不允许
同源策略
同源策略限制了从同一个源加载的文档或脚本如何与来自另一个源的资源进行交互。这是一个用于隔离潜在恶意文件的重要安全机制。浏览器的同源策略的目的就是为了防止XSS
,CSRF
等恶意攻击。
同源策略的交互方式有三种:
- 通常允许跨域写操作,例如链接,重定向等;
- 通常允许跨域嵌套资源,例如
img
,script
标签等; - 通常不允许跨域读操作。
跨域解决方案
跨域资源共享(CORS)
CORS
(Cross-Origin Resource Sharing
)跨域资源共享,定义了必须在访问跨域资源时,浏览器与服务器应该如何沟通。CORS
背后的基本思想就是使用自定义的HTTP
头部让浏览器与服务器进行沟通,从而决定请求或响应是应该成功还是失败。详细解释
<script type="text/javascript">
var xhr = new XMLHttpRequest();
xhr.open("GET", "/example",true);
xhr.send();
</script>
以上的/example
是相对路径,如果我们要使用CORS
,相关Ajax
代码可能如下所示:
<script type="text/javascript">
var xhr = new XMLHttpRequest();
xhr.open("GET", "http://xxx.com/example/",true);
xhr.send();
</script>
代码与之前的区别就在于相对路径换成了其他域的绝对路径,也就是你要跨域访问的接口地址。
服务器端对于CORS
的支持,主要就是通过设置Access-Control-Allow-Origin
来进行的。如果浏览器检测到相应的设置,就可以允许Ajax
进行跨域的访问。
通过document.domain跨域
不同的框架之间是可以获取window
对象的,但却无法获取相应的属性和方法。比如,有一个页面,它的地址是www.example.cn/a.html
, 在这个页面里面有一个iframe
,它的src
是example.cn/b.html
, 很显然,这个页面与它里面的iframe
框架是不同域的,所以我们是无法通过在页面中书写js
代码来获取iframe
中的东西的:
<script type="text/javascript">
function test(){
var iframe = document.getElementById('ifame');
var win = iframe.contentWindow;//可以获取到iframe里的window对象,但该window对象的属性和方法几乎是不可用的
var doc = win.document;//这里获取不到iframe里的document对象
var name = win.name;//这里同样获取不到window对象的name属性
}
</script>
<iframe id = "iframe" src="http://example.cn/b.html" onload = "test()"></iframe>
在页面
example.cn/a.html
中也设置 document.domain。
<iframe id = "iframe" src="http://example.cn/b.html" onload = "test()"></iframe>
<script type="text/javascript">
document.domain = 'example.cn';//设置成主域
function test(){
alert(document.getElementById('iframe').contentWindow);//contentWindow 可取得子窗口的 window 对象
}
</script>
在页面
www.example.cn/b.html
中设置 document.domain。
<script type="text/javascript">
document.domain = 'example.cn';//在iframe载入这个页面也设置document.domain,使之与主页面的document.domain相同
</script>
修改 document.domain 的方法只适用于
不同子域
的框架间的交互。
JSONP
由于浏览器同源策略是允许script
标签这样的跨域资源嵌套的,所以script
标签的资源不受同源策略的限制。JSONP
的解决方案就是通过script
标签进行跨域请求。
- 前端设置好回调函数,并把回调函数当做请求
url
携带的参数; - 后端接受到请求之后,返回回调函数名和需要的数据;
- 后端响应并返回数据后,返回的数据传入到回调函数中并执行。
<!-- 通过原生使用 script 标签 -->
<script>
function jsonpCallback(data) {
alert('获取到的数据了,打开控制台瞧瞧');
console.log(data);
}
</script>
<script src="http://127.0.0.1:3000?callback=jsonpCallback"></script>
也可以使用
AJAX GET
请求方式来跨域请求(axios GET
方式跨域同理)。
<!-- AJAX GET 请求 -->
<script>
function jsonpCallback(data) {
alert('获取到的数据了,打开控制台瞧瞧');
console.log(data);
}
$.ajax({
type: 'GET', // 必须是 GET 请求
url: 'http://127.0.0.1:3000',
dataType: 'jsonp', // 设置为 jsonp 类型
jsonpCallback: 'jsonpCallback' // 设置回调函数
})
</script>
<script type="text/javascript">
$.getJSON('http://example.com/data.php?callback=?,function(jsondata)'){
//处理获得的json数据
});
</script>
jquery
会自动生成一个全局函数来替换callback=?
中的问号,之后获取到数据后又会自动销毁,实际上就是起一个临时代理函数的作用。$.getJSON
方法会自动判断是否跨域,不跨域的话,就调用普通的ajax
方法;跨域的话,则会以异步加载js
文件的形式来调用JSONP
的回调函数。
JSONP的优缺点
- 它不像
XMLHttpRequest
对象实现的Ajax
请求那样受到同源策略的限制;它的兼容性更好,在更加古老的浏览器中都可以运行,不需要XMLHttpRequest
或ActiveX
的支持;并且在请求完毕后可以通过调用callback
的方式回传结果。 - 它只支持
GET
请求而不支持POST
等其它类型的HTTP
请求;它只支持跨域HTTP
请求这种情况,不能解决不同域的两个页面之间如何进行JavaScript
调用的问题。
通过location.hash跨域
iframe
是 html
的一个标签,可以在网页中创建内联框架,有个 src
属性(指向文件地址,html
、php
等)可以选择内联框架的内容。
window.name
(一般在 js
代码里出现)的值不是一个普通的全局变量,而是当前窗口的名字,这里要注意的是每个iframe
都有包裹它的window
,而这个window
是top window
的子窗口,而它自然也有window.name
的属性,window.name
属性的神奇之处在于name
值在不同的页面(甚至不同域名)加载后依旧存在(如果没修改则值不会变化),并且可以支持非常长的name
值(2MB)。
因为父窗口可以对
iframe
进行URL
读写,iframe
也可以读写父窗口的URL
,URL
有一部分被称为hash
,就是#
号及其后面的字符,它一般用于浏览器锚点定位,Server
端并不关心这部分,应该说HTTP
请求过程中不会携带hash
,所以这部分的修改不会产生HTTP
请求,但是会产生浏览器历史记录。此方法的原理就是改变URL
的hash
部分来进行双向通信。
每个window
通过改变其他window
的location
来发送消息(由于两个页面不在同一个域下IE
、Chrome
不允许修改parent.location.hash
的值,所以要借助于父窗口域名下的一个代理iframe
),并通过监听自己的URL
的变化来接收消息。
这个方式的通信会造成一些不必要的浏览器历史记录,而且有些浏览器不支持onhashchange
事件,需要轮询来获知URL
的改变,最后,这样做也存在缺点,诸如数据直接暴露在了URL
中,数据容量和类型都有限等。下面举例说明:
假如父页面是baidu.com/a.html
,iframe
嵌入的页面为google.com/b.html
(此处省略了域名等url
属性),要实现此两个页面间的通信可以通过以下方法。
-
a.html
传送数据到b.html
-
a.html
下修改iframe
的src
为google.com/b.html#paco
; -
b.html
监听到url
发生变化,触发相应操作;
-
-
b.html
传送数据到a.html
,由于两个页面不在同一个域下IE
、Chrome
不允许修改parent.location.hash
的值,所以要借助于父窗口域名下的一个代理iframe
-
b.html
下创建一个隐藏的iframe
,此iframe
的src
是baidu.com
域下的,并挂上要传送的hash
数据,如src="[www.baidu.com/proxy.html#…](https://link.juejin.im?target=http%3A%2F%2Fwww.baidu.com%2Fproxy.html%23data)"
; -
proxy.html
监听到url
发生变化,修改a.html
的url
(因为a.html
和proxy.html
同域,所以proxy.html
可修改a.html
的url hash
; -
a.html
监听到url
发生变化,触发相应操作。
-
b.html
页面的关键代码如下:
try {
parent.location.hash = 'data';
} catch (e) {
// ie、chrome的安全机制无法修改parent.location.hash,
var ifrproxy = document.createElement('iframe');
ifrproxy.style.display = 'none';
ifrproxy.src = "http://www.baidu.com/proxy.html#data";
document.body.appendChild(ifrproxy);
}
proxy.html
页面的关键代码如下 :
// 因为parent.parent(即baidu.com/a.html)和baidu.com/proxy.html属于同一个域,所以可以改变其location.hash的值
parent.parent.location.hash = self.location.hash.substring(1);
通过HTML5的postMessage方法跨域
高级浏览器Internet Explorer 8+
,chrome
,Firefox
, Opera
和 Safari
都将支持这个功能。这个功能主要包括接受信息的message
事件和发送消息的postMessage
方法。比如example.cn
域的 A 页面通过iframe
嵌入了一个google.com
域的 B 页面,可以通过以下方法实现 A 和 B 的通信。
A 页面通过postMessage
方法发送消息:
window.onload = function() {
var ifr = document.getElementById('ifr');
var targetOrigin = "http://www.google.com";
ifr.contentWindow.postMessage('hello world!', targetOrigin);
};
postMessage
的使用方法:
otherWindow.postMessage(message, targetOrigin);
-
otherWindow
:指目标窗口,也就是给哪个window
发消息,是window.frames
属性的成员或者由window.open
方法创建的窗口; -
message
: 是要发送的消息,类型为String
、Object
(IE8
、9
不支持); -
targetOrigin
: 是限定消息接收范围,不限制请使用*
。
B 页面通过message
事件监听并接受消息:
var onmessage = function (event) {
var data = event.data;//消息
var origin = event.origin;//消息来源地址
var source = event.source;//源Window对象
if(origin=="http://www.baidu.com"){
console.log(data);//hello world!
}
};
if (typeof window.addEventListener != 'undefined') {
window.addEventListener('message', onmessage, false);
} else if (typeof window.attachEvent != 'undefined') {
// for ie
window.attachEvent('onmessage', onmessage);
}