跨域请求资源的方法有哪些?

什么事跨域?

由于浏览器同源策略,凡是发送请求url的协议、域名、端口三者之间任意一个与当前页面地址不同即为跨域,以下是存在跨域的情况:

1、网络协议不同,如https协议访问http协议

2、端口不同,如80端口访问8081端口

3、域名不同,如facebook.com访问baidu.com

4、子域名不同,如abc.123.com访问def.123.com

1、通过JSONP跨域

通常为了减轻web服务器的负载,我们把js, css, img等静态资源分离到另一个独立域名的服务器上,在html页面中在通过相应的标签从不同域名下加载静态资源,而被浏览器允许;所以我们可以通过动态创建script,再请求一个带参数的网址实现跨域通信(利用script标签不受同源策略的限制的特性)

一、原生实现:

    var script = document.createElement('script');

    script.type = 'text/javascript';

    // 传参一个回调函数名给后端,方便后端返回时执行这个在前端定义的回调函数

    script.src = 'http://www.domain2.com:8080/login?user=admin&callback=handleCallback';

    document.head.appendChild(script);

    // 回调执行函数

    function handleCallback(res) {

        alert(JSON.stringify(res));

    }

</script>

2) jquery ajax:

$.ajax({

    url:'http://www.domain2.com:8080/login',

    type:'get',

    dataType:'jsonp',// 请求方式为jsonp

    jsonpCallback:"handleCallback",// 自定义回调函数名

    data: {}

});

jsonp缺点:只能实现get一种请求。

二、postMessage跨域

postMessage是HTML5 XMLHttpRequest Level 2中的API,且是为数不多可以跨域操作的window属性之一,它可用于解决以下方面的问题:

a.) 页面和其打开的新窗口的数据传递

b.) 多窗口之间消息传递

c.) 页面与嵌套的iframe消息传递

d.) 上面三个场景的跨域数据传递

用法:postMessage(data,origin)方法接受两个参数

data: html5规范支持任意基本类型或可复制的对象,但部分浏览器只支持字符串,所以传参时最好用JSON.stringify()序列化。

origin: 协议+主机+端口号,也可以设置为"*",表示可以传递给任意窗口,如果要指定和当前窗口同源的话设置为"/"。

1.)a.html:(http://www.domain1.com/a.html)

<iframe id="iframe" src="http://www.domain2.com/b.html" style="display:none;"></iframe>

<script>     

    var iframe = document.getElementById('iframe');

    iframe.onload = function() {

        var data = {

            name: 'aym'

        };

        // 向domain2传送跨域数据

        iframe.contentWindow.postMessage(JSON.stringify(data), 'http://www.domain2.com');

    };

    // 接受domain2返回数据

    window.addEventListener('message', function(e) {

        alert('data from domain2 ---> ' + e.data);

    }, false);

</script>

2.)b.html:(http://www.domain2.com/b.html)

<script>

    // 接收domain1的数据

    window.addEventListener('message', function(e) {

        alert('data from domain1 ---> ' + e.data);

        var data = JSON.parse(e.data);

        if (data) {

            data.number = 16;

            // 处理后再发回domain1

            window.parent.postMessage(JSON.stringify(data), 'http://www.domain1.com');

        }

    }, false);

</script>

三、 跨域资源共享(CORS)

普通跨域请求:只服务端设置Access-Control-Allow-Origin即可,前端无须设置,若要带cookie请求:前后端都需要设置。

需注意的是:由于同源策略的限制,所读取的cookie为跨域请求接口所在域的cookie,而非当前页。如果想实现当前页cookie的写入,可参考下文:七、nginx反向代理中设置proxy_cookie_domain 和 八、NodeJs中间件代理中cookieDomainRewrite参数的设置。

目前,所有浏览器都支持该功能(IE8+:IE8/9需要使用XDomainRequest对象来支持CORS)),CORS也已经成为主流的跨域解决方案。

1.)原生ajax

// 前端设置是否带cookie

xhr.withCredentials =true;

示例代码:

varxhr =newXMLHttpRequest();// IE8/9需用window.XDomainRequest兼容

// 前端设置是否带cookie

xhr.withCredentials=true;

xhr.open('post','http://www.domain2.com:8080/login',true);

xhr.setRequestHeader('Content-Type','application/x-www-form-urlencoded');

xhr.send('user=admin');

xhr.onreadystatechange =function(){

if(xhr.readyState ==4&& xhr.status ==200){        

alert(xhr.responseText);    

}}

2.)jQuery ajax

$.ajax({

    ...   xhrFields: {withCredentials:true // 前端设置是否带cookie

},

crossDomain:true, // 会让请求头中包含跨域的额外信息,但不会含cookie

...});

3.)vue框架

a.) axios设置:

axios.defaults.withCredentials = true

b.) vue-resource设置:

Vue.http.options.credentials = true

四、WebSocket协议跨域

WebSocket protocol是HTML5一种新的协议。它实现了浏览器与服务器全双工通信,同时允许跨域通讯,是server push技术的一种很好的实现。

原生WebSocket API使用起来不太方便,我们使用Socket.io,它很好地封装了webSocket接口,提供了更简单、灵活的接口,也对不支持webSocket的浏览器提供了向下兼容。

1.)前端代码:

if ("WebSocket" in window) { 

 alert("您的浏览器支持 WebSocket!");

              // 打开一个 web socket  

            var ws = new WebSocket("ws://localhost:9998/echo");                

              ws.onopen = function()   {                 

                 // Web Socket 已连接上,使用 send() 方法发送数据 

                 ws.send("发送数据"); 

                 alert("数据发送中...");

              };                

              ws.onmessage = function (evt)

              {

                  var received_msg = evt.data; 

                 alert("数据已接收...");

              };                

              ws.onclose = function()              {

                  // 关闭 websocket 

                 alert("连接已关闭...");

              }; 

           }

仅作为常用笔记

借鉴地址https://segmentfault.com/a/1190000011145364

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

推荐阅读更多精彩内容

  • 什么是跨域 跨域,是指浏览器不能执行其他网站的脚本。它是由浏览器的同源策略造成的,是浏览器对JavaScript实...
    HeroXin阅读 836评论 0 4
  • 什么是跨域 跨域,是指浏览器不能执行其他网站的脚本。它是由浏览器的同源策略造成的,是浏览器对JavaScript实...
    他方l阅读 1,064评论 0 2
  • 什么是跨域 跨域,是指浏览器不能执行其他网站的脚本。它是由浏览器的同源策略造成的,是浏览器对JavaScript实...
    Yaoxue9阅读 1,299评论 0 6
  • 一 、JSONP 利用script标签的src属性天然可以跨域的特点,在跨域脚本中可以直接回调当前脚本的函数 va...
    海子小夜曲阅读 301评论 0 0
  • 乐观和悲观,是对事物两种截然不同的看法,是两种不同的心态。 我相信,我们绝大多数人还是愿意循着乐观的足迹到人世间走...
    百合鸟飞阅读 332评论 2 3