配色: 字号:
前端跨域问题的几种解决方案
2017-01-21 | 阅:  转:  |  分享 
  
前端跨域问题的几种解决方案

前端跨域问题



一:同源策略

1.what''sthis

所谓同源是指,域名,协议,端口相同。当浏览器运行一个JS脚本时会进行同源检测,如果不同源是不能执行的。

2.源继承

来自about:blank,javascript:和data:URLs中的内容,继承了将其载入的文档所指定的源,因为它们的URL本身未指定任何关于自身源的信息。

3.变更源

变更源可以实现基础域相同的不同页面的跨域问题。

如:a.baidu.com/index.html通过iframe引入b.baidu.com/index.html,但是a中的JS是不可以操作b中的内容的,但是可以通过修改源来实现。需要在a和b中都修改domain,即document.domain=''baicu.com''

注意:document.doamin的修改必须是当前域或者当前域的基础域,如在a中document.domain=''b.baidu.com''是报错的

二:跨域方案



共有几种解决方案:



(1)document.domain+iframe



(2)动态创建script



(3)window.name+iframe



(4)window.postMessage



(5)CORS



(6)JSONP



(7)nginx代理





1.document.domain+iframe



这种方式就是上面说的变更源

在a.name.com/a.html中





document.domain=''a.com'';



varifr=document.createElement(''iframe'');

ifr.src=''http://b.name.com/b.html'';

ifr.display=none;

document.body.appendChild(ifr);



ifr.onload=function(){

vardoc=ifr.contentDocument||ifr.contentWindow.document;

//在这里操作doc,也就是b.html

ifr.onload=null;

};

在b.name.com/b.html中



1

document.domain=''name.com'';

2.动态创建script



因为script标签不受同源策略的限制





functionloadScript(url,func){

varhead=document.head||document.getElementByTagName(''head'')[0];

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

script.src=url;



script.onload=script.onreadystatechange=function(){

if(!this.readyState||this.readyState==''loaded''||this.readyState==''complete''){

func();

script.onload=script.onreadystatechange=null;

}

};



head.insertBefore(script,script[0]);

}

window.baidu={

sug:function(data){

console.log(data);

}

}

loadScript(''https://www.baidu.com'',function(){console.log(''loaded'')});

3.window.name+iframe



window对象有个name属性,该属性有个特征:即在一个窗口(window)的生命周期内,窗口载入的所有的页面都是共享一个window.name的,每个页面对window.name都有读写的权限,window.name是持久存在一个窗口载入过的所有页面中的,并不会因新页面的载入而进行重置



a.com/a.html



























4.postMessage(HTML5中的XMLHttpRequestLevel2中的API)



window.postMessage(message,targetOrigin)方法是html5新引进的特性,可以使用它来向其它的window对象发送消息,无论这个window对象是属于同源或不同源,目前IE8+、FireFox、Chrome、Opera等浏览器都已经支持window.postMessage方法。



调用postMessage方法的window对象是指要接收消息的那一个window对象,该方法的第一个参数message为要发送的消息,类型只能为字符串;第二个参数targetOrigin用来限定接收消息的那个window对象所在的域,如果不想限定域,可以使用通配符。



需要接收消息的window对象,可是通过监听自身的message事件来获取传过来的消息,消息内容储存在该事件对象的data属性中。



上面所说的向其他window对象发送消息,其实就是指一个页面有几个框架的那种情况,因为每一个框架都有一个window对象。在讨论第二种方法的时候,我们说过,不同域的框架间是可以获取到对方的window对象的,而且也可以使用window.postMessage这个方法。下面看一个简单的示例,有两个页面



a.com/index.html



























b.com/index.html







5.CORS(Cross-OriginResourceSharing)



跨源资源共享(CORS)是通过客户端+服务端协作声明的方式来确保请求安全的。服务端会在HTTP请求头中增加一系列HTTP请求参数(例如Access-Control-Allow-Origin等),来限制哪些域的请求和哪些请求类型可以接受,而客户端在发起请求时必须声明自己的源(Orgin),否则服务器将不予处理,如果客户端不作声明,请求甚至会被浏览器直接拦截都到不了服务端。



前端:





functiongetHello(){

varxhr=newXMLHttpRequest();

xhr.open("post","https://b.example.com/Test.ashx",true);

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



xhr.onreadystatechange=function(){

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

varresponseText=xhr.responseText;

console.info(responseText);

}

}

xhr.send();

}

服务端:(https://b.example.com/Test.ashx)



1

header(''Access-Control-Allow-Origin:'')

也可以指定具体的来源



6.JSONP





functionhandleResponse(response){

console.log(''Theresponseddatais:''+response.data);

}

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

script.src=''http://www.baidu.com/json/?callback=handleResponse'';

document.body.insertBewww.baiyuewang.netfore(script,document.body.firstChild);

7.Nginx反向代理



前端调用的服务/apis/xxxx/xxxx和当前页是同源的,nginx来做一个代理到想要的地方,来实现跨域



nginx.conf配置一个反向代理路径





location/apis{

rewrite^.+apis/?(.)$/$1break;

includeuwsgi_params;

proxy_passhttp://www.baicu.com/xxxx

}

献花(0)
+1
(本文系thedust79首藏)