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

什么事跨域?

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

    }

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)

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

三、 跨域资源共享(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

你可能感兴趣的:(跨域请求资源的方法有哪些?)