跨域的几种方法

同源策略

浏览器出于安全方面的考虑,为了保证用户信息的安全,防止恶意的网站窃取数据。只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情况下,不能读写对方的资源 。
而同源指的是:
同协议:如都是http或者https
同域名:如都是http://jirengu.com/ahttp://jirengu.com/b
同端口:如都是80端口
所以只要协议、域名、端口有任何一个不同,都被当作是不同的域。

跨域的方法

1 JSONP

JSONP的全称:JSON with Padding,它是资料格式 JSON 的一种“使用模式”,可以让网页从别的网域要资料。
JSONP由两部分组成:回调函数和数据。回调函数是当响应到来时应该在页面中调用的函数,而数据就是传入回调函数中的JSON数据。

在HTML中通过添加一个<script>元素,向服务器请求JSON数据,并定义回调函数

直接设置<script>
<script src="http://example.com/getNews?callback=foo"></script>   //foo是回调函数 

还可以通过DOM来创建<script>
  $('.change').addEventListener('click', function(){      //绑定click事件
    var script = document.createElement('script');          //创建<script>元素
    script.src ="http:example.com/getNews?callback=foo";        //设置src属性,形成url
    document.head.appendChild(script);     // 添加<script>
    document.head.removeChild(script);     //删除<script>
  })

function foo(){       //定义回调函数
  code      //处理得到的数据
}

服务器端也需要进行相应的设置,回传数据

app.get('/getNews', function(req, res){

    ...


    var cb = req.query.callback;
    if(cb){
        res.send(cb + '('+ JSON.stringify(data) + ')');  //如果有回调函数,返回 foo(JSON数据)
    }else{
        res.send(data);   //返回默认数据
    }   
    
})

服务器收到请求后,将数据放在一个指定名字的回调函数里传回来。由于<script>元素请求的脚本,直接作为代码运行。这时,只要浏览器定义了回调函数,该函数就会立即调用。

优缺点

优点:JSONP不像XMLHttpRequest对象实现的Ajax请求那样受到同源策略的限制;它的兼容性更好,在更加古老的浏览器中都可以运行,不需要XMLHttpRequest或ActiveX的支持;并且在请求完毕后可以通过调用callback的方式回传结果。

缺点:JSONP只支持GET请求而不支持POST等其它类型的HTTP请求;它只支持跨域HTTP请求这种情况,不能解决不同域的两个页面之间如何进行JavaScript调用的问题。

2 CORS

CORS全称是"跨域资源共享"(Cross-origin resource sharing),定义了必须在访问跨域资源时,浏览器与服务器应该如何沟通,支持IE10以上的浏览器。 实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。

服务器端设置Access-Control-Allow-Origin

res.header("Access-Control-Allow-Origin", "http://a.jrg.com:8080");   //接受'http://a.jrg.com:8080'的请求
res.header("Access-Control-Allow-Origin", "*");     //接受任何域名的请求

此外,CORS请求分成两类:简单请求和非简单请求。浏览器对这两种请求的处理,是不一样的。具体参考阮老师的博客

3 降域

不同的框架之间是可以获取window对象的,但却无法获取相应的属性和方法。比如以下a.html中有个b.html子页面,因为它们域名不同,两者之间无法进行交互。我们可以通过设置document.domain,让两者的域名相同。

//a.html
<input type="text" placeholder="http://a.jrg.com:8080/a.html">
<iframe src="http://b.jrg.com:8080/b.html" frameborder="0" ></iframe>


<script>
document.querySelector('.main input').addEventListener('input', function(){
  console.log(this.value);
  window.frames[0].document.querySelector('input').value = this.value;
})
document.domain = "jrg.com"  //设置主域名为jrg.com
</script>
//b.html
<input id="input" type="text"  placeholder="http://b.jrg.com:8080/b.html">

<script>
document.querySelector('#input').addEventListener('input', function(){
  window.parent.document.querySelector('input').value = this.value;
})
document.domain = 'jrg.com';  //设置主域名为jrg.com
</script>

document.domain的设置是有限制的,我们只能把document.domain设置成自身或更高一级的父域,且主域必须相同。修改document.domain的方法只适用于不同子域的框架间的交互。

4 postMessage

HTML5为了解决跨域窗口的通信问题,引入了一个全新的API:跨文档通信 API(Cross-document messaging)。

otherWindow.postMessage(message, targetOrigin, [transfer]);  //postMessage方法的第一个参数是具体的信息内容,第二个参数是接收消息的窗口的源(origin),即"协议 + 域名 + 端口"。也可以设为*,表示不限制域名,向所有窗口发送。
//父元素a.html
<iframe src="http://b.jrg.com:8080/b.html" frameborder="0" ></iframe>

window.frames[0].postMessage('Hello World!', 'http://b.jrg.com:8080/b.html');   //向b.html发送'Hello World!'

window.addEventListener('message', function(e) {
  console.log(e.data);   //监听到并打印出'Nice to see you'
},false);

//子元素b.html
window.parent.postMessage('Nice to see you', 'http://a.jrg.com:8080/a.html'); //向a.html发送'Nice to see you'

window.addEventListener('message', function(e) {
  console.log(e.data);   //监听到并打印出'Hello World!'
},false);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • 1. AJAX AJAX(Asynchronous JavaScript and XML),意思就是用JavaSc...
    公子七阅读 4,967评论 0 5
  • 什么是跨域 浏览器出于安全方面的考虑,只允许客户端与本域(同协议、同域名、同端口,三者缺一不可)下的接口交互。不同...
    七里之境阅读 1,318评论 0 0
  • 什么是跨域 说到跨域必须先解释什么是同源策略,它是由Netscape提出的一个著名的安全策略。浏览器出于安全方面的...
    8d2855a6c5d0阅读 1,315评论 0 0
  • CORS 实现CORS需要浏览器和服务器的共同协作。对于前端开发者来说,CORS通信与同源的AJAX在代码方面没有...
    饥人谷_有点热阅读 1,265评论 0 2
  • 概念:只要协议、域名、端口有任何一个不同,都被当作是不同的域。 所有具有 src 属性的HTML标签都可以跨域 原...
    jeffAAA阅读 2,495评论 0 0