跨域(补充)

2018-06-05  本文已影响17人  Ertsul

document.domain

Cookie是服务器写入浏览器的信息,只有同源的页面才能共享。但是,如果两个页面的一级域名一样,二级域名不一样,则允许通过将两个页面的document.domain共享资源。

域名的级别

document.domain + Cookie

举例说明,非同源的两个页面A和B,A的地址为[http://a.example.com/a.html,B的地址为:http://b.example.com/b.html。这时,只要设强制置 A 和 B 的 document.domain 相同,就可以实现跨域请求了。

// A
document.domain = 'example.com'
document.cookie = 'Hello, I save a cookie.'
// B
document.domain = 'example.com'
document.cookie = 'Hello, I save a cookie.'

上面的例子,通过将AB两个页面设置相同的document.cookie,然后在A页面设置一个cookie,在B页面就可以获取A页面的cookie。

document.domain + iframe

这种方法的实现原理跟document.domain+Cookie是一样的,都是将两个不同的页面设置相同的document.domain,不同的是不需要通过设置cookie获取数据,而是通过父子两个窗口进行通信。

举个例子,A和B页面非同源,A页面内嵌一个iframe,加载这B页面。A的地址为:http://a.example.com/a.html,B的地址为:http://b.example.com/b.html

// A
<iframe id="iframe" src="http://b.example.com/b.html"></iframe>
<script>
 document.domain = 'example.com';
 var hi = 'hello';
</script>

// B
<script>
 document.domain = 'example.com';
 // 获取父窗口中变量
 alert(window.parent.hi);
</script>

location.hash + iframe

什么是location.hash?也就是片段标识符(fragment identifier)?指的是在URL的后面添加#号,这个符号后面的值就是location.hash。这样呢,就可以在不同源的两个页面(iframe父子页面)的后面添加hash值,进行监听。

举例:AB两个页面非同源,A页面内嵌了一个非同源地址的B页面。【 A将数据通过hash传递给B 】

// A页面
<iframe src="A的地址"></iframe>
<input type="text" name="" value="">
<button type="button">btnbtn</button>
<script>
 window.onload = function () { 
 document.getElementsByTagName('button')[0].addEventListener('click', function () { 
 let data = document.getElementsByTagName('input')[0].value
 document.getElementsByTagName('iframe')[0].setAttribute('src', 'A的地址#' + data)
 })
 }
</script>

// B页面
<script>
 window.onhashchange = function () { 
 console.log(window.location.hash);
 }
</script>

分析:A通过hash将数据传递给B,B页面通过window.onhashchange进行监听,通过window.location.hash获取数据。

window.name

浏览器窗口有window.name属性。这个属性的最大特点是,无论是否同源,只要在同一个窗口里,前一个网页设置了这个属性,后一个网页可以读取它。name值在不同的页面(甚至不同域名)加载后依旧存在,并且可以支持非常长的name值(2MB)。
实现思路:A和B为非同源的页面。在A中打开一个新窗口B,在B中将信息写入window.name属性,然后,B跳转到与A同源的另一个页面C(AC同源,C页面可以说是代理页面,可以设置为空白页),这样A窗口就可以获取B窗口的window.name属性值了。

// 在A打开子窗口B,并在B中写入window属性
window.name = data
// B跳转到C(AC同源)
location = '...'
// A读取B的window.name
let data = document.getElementById('myFrame').contentWindow.name

postMessage实现跨域

postMessage是html5提供的API,即:跨文档通信API(Cross-document messaging),是为数不多的可以跨域操作的window属性之一。这个API为window对象新增加了一个window.postMessage方法,允许跨窗口通信,无论这些窗口是否同源。
这个API可以解决这些问题:

postMessage(data, origin)

这个方法会接受两个参数,这两个参数分别是:

举一个例子:

// A页面
let newPage = window.open('http://b.example.com', 'title')
newPage.postMessage('hello, I am a page.', 'http://b.example.com')
// B页面监听
window.addEventListener('message', function (e) {
 e.data
}, false)
// B向A发送消息(子窗口想父窗口发送消息)
window.opener.postMessage('Nice to see you', 'http://a.example.com');

这样子,也可以实现跨域的访问。当然,子窗口也可以发送消息给父窗口,父子窗口的监听方法一样。

message事件对象主要有三个属性:

CORS

CORS是跨域资源分享(Cross-Origin Resource Sharing)的缩写。跟JSONP相比较,JSONP只能实现GET方法的请求,而CORS则允许任何类型的请求。

上一篇下一篇

猜你喜欢

热点阅读