美文网首页
跨域请求资源的方法有哪些?

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

作者: 我有一条小黑狗 | 来源:发表于2019-07-02 16:54 被阅读0次

    什么事跨域?

    由于浏览器同源策略,凡是发送请求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

    相关文章

      网友评论

          本文标题:跨域请求资源的方法有哪些?

          本文链接:https://www.haomeiwen.com/subject/oedonqtx.html