什么是跨域?跨域解决方法
一、为什么会出现跨域问题
出于浏览器的同源策略限制。同源策略(Sameoriginpolicy)是一种约定,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,则浏览器的正常功能可能都会受到影响。可以说Web是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现。同源策略会阻止一个域的javascript脚本和另外一个域的内容进行交互。所谓同源(即指在同一个域)就是两个页面具有相同的协议(protocol),主机(host)和端口号(port)
二、什么是跨域
当一个请求url的协议、域名、端口三者之间任意一个与当前页面url不同即为跨域
当前页面url被请求页面url是否跨域原因
(协议、域名、端口号相同)
(http/https)
(test/baidu)
(www/blog)
(8080/7001)
三、非同源限制
【1】无法读取非同源网页的Cookie、LocalStorage和IndexedDB
【2】无法接触非同源网页的DOM
【3】无法向非同源地址发送AJAX请求
四、跨域解决方法
【1】设置解决无法读取非同源网页的Cookie问题
因为浏览器是通过属性来检查两个页面是否同源,因此只要通过设置相同的,两个页面就可以共享Cookie(此方案仅限主域相同,子域不同的跨域应用场景。)
//两个页面都设置
='';
【2】跨文档通信API:()
调用postMessage方法实现父窗口(子窗口同样可以通过该方法发送消息给父窗口)
它可用于解决以下方面的问题:
页面和其打开的新窗口的数据传递
多窗口之间消息传递
页面与嵌套的iframe消息传递
上面三个场景的跨域数据传递
//父窗口打开一个子窗口
varopenWindow=('','title');
//父窗口向子窗口发消息(第一个参数代表发送的内容,第二个参数代表接收消息窗口的url)
('Nicetomeetyou!','');
调用message事件,监听对方发送的消息
//监听message消息
('message',function(e){
();//发送消息的窗口
();//消息发向的网址
();//发送的消息
},false);
【3】JSONP
JSONP是服务器与客户端跨源通信的常用方法。最大特点就是简单适用,兼容性好(兼容低版本IE),缺点是只支持get请求,不支持post请求。
核心思想:网页通过添加一个script元素,向服务器请求JSON数据,服务器收到请求后,将数据放在一个指定名字的回调函数的参数位置传回来。
①原生实现:
scriptsrc=""/script
//向服务器发出请求,该请求的查询字符串有一个callback参数,用来指定回调函数的名字
//处理服务器返回回调函数的数据
scripttype="text/javascript"
functiondosomething(res){
//处理获得的数据
()
}
/script
②jQueryajax:
$.ajax({
url:'',
type:'get',
dataType:'jsonp',//请求方式为jsonp
jsonpCallback:"handleCallback",//自定义回调函数名
data:{}
});
③
this.$('',{
params:{},
jsonp:'handleCallback'
}).then((res)={
(res);
})
【4】CORS
CORS是跨域资源分享(Cross-OriginResourceSharing)的缩写。它是W3C标准,属于跨源AJAX请求的根本解决方法。
1、普通跨域请求:只需服务器端设置Access-Control-Allow-Origin
2、带cookie跨域请求:前后端都需要进行设置
【前端设置】根据字段判断是否带有cookie
①原生ajax
varxhr=newXMLHttpRequest();//IE8/9需用兼容
//前端设置是否带cookie
=true;
('post','',true);
('Content-Type','application/x-www-form-urlencoded');
('user=admin');
=function(){
if(==4==200){
alert();
}
};
②jQueryajax
$.ajax({
url:'',
type:'get',
data:{},
xhrFields:{
withCredentials:true//前端设置是否带cookie
},
crossDomain:true,//会让请求头中包含跨域的额外信息,但不会含cookie
});
③vue-resource
=true
④axios
=true
【服务端设置】
服务器端对于CORS的支持,主要是通过设置Access-Control-Allow-Origin来进行的。如果浏览器检测到相应的设置,就可以允许Ajax进行跨域的访问。
①Java后台
/*
*导入包:;
*接口参数中定义:HttpServletResponseresponse
*/
//允许跨域访问的域名:若有端口需写全(协议+域名+端口),若没有端口末尾不用加'/'
("Access-Control-Allow-Origin","");
//允许前端带认证cookie:启用此项后,上面的域名不能为'*',必须指定具体的域名,否则浏览器会提示
("Access-Control-Allow-Credentials","true");
//提示OPTIONS预检时,后端需要设置的两个常用自定义头
("Access-Control-Allow-Headers","Content-Type,X-Requested-With");
②Nodejs后台
varhttp=require('http');
varserver=();
varqs=require('querystring');
('request',function(req,res){
varpostData='';
//数据块接收中
('data',function(chunk){
postData+=chunk;
});
//数据接收完毕
('',function(){
postData=(postData);
//跨域后台设置
(200,{
'Access-Control-Allow-Credentials':'true',//后端允许发送Cookie
'Access-Control-Allow-Origin':'',//允许访问的域(协议+域名+端口)
/*
*此处设置的cookie还是domain2的而非domain1,因为后端也不能跨域写cookie(nginx反向代理可以实现),
*但只要domain2中写入一次cookie认证,后面的跨域接口都能从domain2中获取cookie,从而实现所有的接口都能跨域访问
*/
'Set-Cookie':'l=a123456;Path=/;Domain=;HttpOnly'//HttpOnly的作用是让js无法读取cookie
});
((postData));
();
});
});
('8080');
('Serverisrunningatport8080');
③PHP后台
?php
header("Access-Control-Allow-Origin:*");
④Apache需要使用mod_headers模块来激活HTTP头的设置,它默认是激活的。你只需要在Apache配置文件的Directory,Location,Files或VirtualHost的配置里加入以下内容即可
HeadersetAccess-Control-Allow-Origin