JS跨域(Access-Control-Allow-Origin)前后端解決方案詳解
瀏覽器的同源安全策略
同源策略,它是由Netscape提出的一個(gè)著名的安全策略?,F(xiàn)在所有支持JavaScript的瀏覽器都會(huì)使用這個(gè)策略。所謂同源是指,域名,協(xié)議,端口相同。同源策略是瀏覽器的行為,是為了保護(hù)本地?cái)?shù)據(jù)不被JavaScript代碼獲取回來(lái)的數(shù)據(jù)污染,因此攔截的是客戶端發(fā)出的請(qǐng)求回來(lái)的數(shù)據(jù)接收,即請(qǐng)求發(fā)送了,服務(wù)器響應(yīng)了,但是無(wú)法被瀏覽器接收。
同源:協(xié)議 + 域名 + 端口。所以,怎么才算跨域呢?
什么是跨域
什么是跨域,簡(jiǎn)單地理解就是因?yàn)镴avaScript同源策略的限制,a.com 域名下的js無(wú)法操作b.com或是c.a.com域名下的對(duì)象。更詳細(xì)的說明可以看下表:
| URL | 說明 | 是否允許通信 |
|---|---|---|
| http://www.a.com/a.js http://www.a.com/b.js | 同一域名下 | 允許 |
| http://www.a.com/lab/a.js http://www.a.com/script/b.js | 同一域名下不同文件夾 | 允許 |
| http://www.a.com:8000/a.js http://www.a.com/b.js | 同一域名,不同端口 | 不允許 |
| http://www.a.com/a.js https://www.a.com/b.js | 同一域名,不同協(xié)議 | 不允許 |
| http://www.a.com/a.js http://70.32.92.74/b.js | 域名和域名對(duì)應(yīng)ip | 不允許 |
| http://www.a.com/a.js http://script.a.com/b.js | 主域相同,子域不同 | 不允許 |
| http://www.a.com/a.js http://a.com/b.js | 同一域名,不同二級(jí)域名(同上) | 不允許(cookie這種情況下也不允許訪問) |
| //www.dbjr.com.cn/a.js http://www.a.com/b.js | 不同域名 | 不允許 |
特別注意兩點(diǎn):
第一,如果是協(xié)議和端口造成的跨域問題“前臺(tái)”是無(wú)能為力的,
第二:在跨域問題上,域僅僅是通過“URL的首部”來(lái)識(shí)別而不會(huì)去嘗試判斷相同的ip地址對(duì)應(yīng)著兩個(gè)域或兩個(gè)域是否在同一個(gè)ip上。
“URL的首部”指window.location.protocol +window.location.host,也可以理解為“Domains, protocols and ports must match”。
一、前端跨域解決方法(JavaScript)
接下來(lái)簡(jiǎn)單地總結(jié)一下在“前臺(tái)”一般處理跨域的辦法
1、document.domain+iframe的設(shè)置
對(duì)于主域相同而子域不同的例子,可以通過設(shè)置document.domain的辦法來(lái)解決。具體的做法是可以在http://www.a.com/a.html和http://script.a.com/b.html兩個(gè)文件中分別加上document.domain = ‘a.com’;然后通過a.html文件中創(chuàng)建一個(gè)iframe,去控制iframe的contentDocument,這樣兩個(gè)js文件之間就可以“交互”了。當(dāng)然這種辦法只能解決主域相同而二級(jí)域名不同的情況,如果你異想天開的把script.a.com的domian設(shè)為alibaba.com那顯然是會(huì)報(bào)錯(cuò)地!代碼如下:
www.a.com上的a.html
document.domain = 'a.com';
var ifr = document.createElement('iframe');
ifr.src = 'http://script.a.com/b.html';
ifr.style.display = 'none';
document.body.appendChild(ifr);
ifr.onload = function(){
var doc = ifr.contentDocument || ifr.contentWindow.document;
// 在這里操縱b.html
alert(doc.getElementsByTagName("h1")[0].childNodes[0].nodeValue);
};script.a.com上的b.html
document.domain = 'a.com';
這種方式適用于{www.jb51.com, jb51.com, script.jb51.com, css.jb51.com}中的任何頁(yè)面相互通信。
備注:某一頁(yè)面的domain默認(rèn)等于window.location.hostname。主域名是不帶www的域名,例如a.com,主域名前面帶前綴的通常都為二級(jí)域名或多級(jí)域名,例如www.a.com其實(shí)是二級(jí)域名。 domain只能設(shè)置為主域名,不可以在b.a.com中將domain設(shè)置為c.a.com。
問題:
1、安全性,當(dāng)一個(gè)站點(diǎn)(b.a.com)被攻擊后,另一個(gè)站點(diǎn)(c.a.com)會(huì)引起安全漏洞。
2、如果一個(gè)頁(yè)面中引入多個(gè)iframe,要想能夠操作所有iframe,必須都得設(shè)置相同domain。
2、動(dòng)態(tài)創(chuàng)建script
雖然瀏覽器默認(rèn)禁止了跨域訪問,但并不禁止在頁(yè)面中引用其他域的JS文件,并可以自由執(zhí)行引入的JS文件中的function(包括操作cookie、Dom等等)。根據(jù)這一點(diǎn),可以方便地通過創(chuàng)建script節(jié)點(diǎn)的方法來(lái)實(shí)現(xiàn)完全跨域的通信。
這里判斷script節(jié)點(diǎn)加載完畢還是蠻有意思的:ie只能通過script的readystatechange屬性,其它瀏覽器是script的load事件。以下是部分判斷script加載完畢的方法。
js.onload = js.onreadystatechange = function() {
if (!this.readyState || this.readyState === 'loaded' || this.readyState === 'complete') {
// callback在此處執(zhí)行
js.onload = js.onreadystatechange = null;
}
};3、利用iframe和location.hash
這個(gè)辦法比較繞,但是可以解決完全跨域情況下的腳步置換問題。原理是利用location.hash來(lái)進(jìn)行傳值。在url: http://a.com#helloword中的‘#helloworld’就是location.hash,改變hash并不會(huì)導(dǎo)致頁(yè)面刷新,所以可以利用hash值來(lái)進(jìn)行數(shù)據(jù)傳遞,當(dāng)然數(shù)據(jù)容量是有限的。假設(shè)域名a.com下的文件cs1.html要和jb51.net域名下的cs2.html傳遞信息,cs1.html首先創(chuàng)建自動(dòng)創(chuàng)建一個(gè)隱藏的iframe,iframe的src指向jb51.net域名下的cs2.html頁(yè)面,這時(shí)的hash值可以做參數(shù)傳遞用。cs2.html響應(yīng)請(qǐng)求后再將通過修改cs1.html的hash值來(lái)傳遞數(shù)據(jù)(由于兩個(gè)頁(yè)面不在同一個(gè)域下IE、Chrome不允許修改parent.location.hash的值,所以要借助于a.com域名下的一個(gè)代理iframe;Firefox可以修改)。同時(shí)在cs1.html上加一個(gè)定時(shí)器,隔一段時(shí)間來(lái)判斷l(xiāng)ocation.hash的值有沒有變化,一點(diǎn)有變化則獲取獲取hash值。代碼如下:
先是a.com下的文件cs1.html文件:
function startRequest(){
var ifr = document.createElement('iframe');
ifr.style.display = 'none';
ifr.src = '//www.dbjr.com.cn/lab/cscript/cs2.html#paramdo';
document.body.appendChild(ifr);
}
function checkHash() {
try {
var data = location.hash ? location.hash.substring(1) : '';
if (console.log) {
console.log('Now the data is '+data);
}
} catch(e) {};
}
setInterval(checkHash, 2000);jb51.net域名下的cs2.html:
//模擬一個(gè)簡(jiǎn)單的參數(shù)處理操作
switch(location.hash){
case '#paramdo':
callBack();
break;
case '#paramset':
//do something……
break;
}
function callBack(){
try {
parent.location.hash = 'somedata';
} catch (e) {
// ie、chrome的安全機(jī)制無(wú)法修改parent.location.hash,
// 所以要利用一個(gè)中間的cnblogs域下的代理iframe
var ifrproxy = document.createElement('iframe');
ifrproxy.style.display = 'none';
ifrproxy.src = 'http://a.com/test/cscript/cs3.html#somedata'; // 注意該文件在"a.com"域下
document.body.appendChild(ifrproxy);
}
}a.com下的域名cs3.html
//因?yàn)閜arent.parent和自身屬于同一個(gè)域,所以可以改變其location.hash的值 parent.parent.location.hash = self.location.hash.substring(1);
當(dāng)然這樣做也存在很多缺點(diǎn),諸如數(shù)據(jù)直接暴露在了url中,數(shù)據(jù)容量和類型都有限等……
4、window.name實(shí)現(xiàn)的跨域數(shù)據(jù)傳輸
文章較長(zhǎng)列在此處不便于閱讀,詳細(xì)請(qǐng)看 window.name實(shí)現(xiàn)的跨域數(shù)據(jù)傳輸。
5、使用HTML5 postMessage
HTML5中最酷的新功能之一就是 跨文檔消息傳輸Cross Document Messaging。下一代瀏覽器都將支持這個(gè)功能:Chrome 2.0+、Internet Explorer 8.0+, Firefox 3.0+, Opera 9.6+, 和 Safari 4.0+ 。 Facebook已經(jīng)使用了這個(gè)功能,用postMessage支持基于web的實(shí)時(shí)消息傳遞。
otherWindow.postMessage(message, targetOrigin);
otherWindow: 對(duì)接收信息頁(yè)面的window的引用??梢允琼?yè)面中iframe的contentWindow屬性;window.open的返回值;通過name或下標(biāo)從window.frames取到的值。
message: 所要發(fā)送的數(shù)據(jù),string類型。
targetOrigin: 用于限制otherWindow,“*”表示不作限制
a.com/index.html中的代碼:
<iframe id="ifr" src="b.com/index.html"></iframe>
<script type="text/javascript">
window.onload = function() {
var ifr = document.getElementById('ifr');
var targetOrigin = 'http://b.com'; // 若寫成'http://b.com/c/proxy.html'效果一樣
// 若寫成'http://c.com'就不會(huì)執(zhí)行postMessage了
ifr.contentWindow.postMessage('I was there!', targetOrigin);
};
</script>b.com/index.html中的代碼:
<script type="text/javascript">
window.addEventListener('message', function(event){
// 通過origin屬性判斷消息來(lái)源地址
if (event.origin == 'http://a.com') {
alert(event.data); // 彈出"I was there!"
alert(event.source); // 對(duì)a.com、index.html中window對(duì)象的引用
// 但由于同源策略,這里event.source不可以訪問window對(duì)象
}
}, false);
</script>二、后臺(tái)跨域解決方案
CORS
這是W3C的標(biāo)準(zhǔn),全稱是"跨域資源共享"(Cross-origin resource sharing)。
//指定允許其他域名訪問 ‘Access-Control-Allow-Origin:http://172.20.0.206'//一般用法(,指定域,動(dòng)態(tài)設(shè)置),3是因?yàn)椴辉试S攜帶認(rèn)證頭和cookies //是否允許后續(xù)請(qǐng)求攜帶認(rèn)證信息(cookies),該值只能是true,否則不返回 ‘Access-Control-Allow-Credentials:true'
上面第一行說到的Access-Control-Allow-Origin有多種設(shè)置方法:
設(shè)置是最簡(jiǎn)單粗暴的,但是服務(wù)器出于安全考慮,肯定不會(huì)這么干,而且,如果是的話,游覽器將不會(huì)發(fā)送cookies,即使你的XHR設(shè)置了withCredentials
指定域,如上圖中的http://172.20.0.206,一般的系統(tǒng)中間都有一個(gè)nginx,所以推薦這種
動(dòng)態(tài)設(shè)置為請(qǐng)求域,多人協(xié)作時(shí),多個(gè)前端對(duì)接一個(gè)后臺(tái),這樣很方便。
Response支持跨域
從上面控制臺(tái)的輸出可以看到,錯(cuò)誤原因是請(qǐng)求的資源(接口)的header中沒有”Access-Control-Allow-Origin“,那我們可以給它加上。在哪加?既然說是請(qǐng)求的資源沒有,那當(dāng)然是在請(qǐng)求的資源上加,也就是服務(wù)端。
@SpringBootApplication
@Configuration
@RestController
public class ApplicationA {
public static void main(String[] args) {
SpringApplication.run(ApplicationA.class, args);
}
@RequestMapping("/test")
public Object test(HttpServletRequest request, HttpServletResponse response) {
// 跨域支持
response.setHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Access-Control-Allow-Methods", "POST,GET,PUT,DELETE");
response.setHeader("Access-Control-Max-Age", "3600");
response.setHeader("Access-Control-Allow-Headers", "*");
response.setHeader("Access-Control-Allow-Credentials", "true");
Map<String, Object> map = new HashMap<>();
map.put("success", true);
map.put("msg", "我來(lái)自服務(wù)端");
return map;
}
}springboot支持跨域
測(cè)試用例是一個(gè)springboot項(xiàng)目,可以用更簡(jiǎn)單的方式。通過一個(gè)繼承了WebMvcConfigurerAdapter的bean,重寫addCorsMappings方法,在方法里配置。
@SpringBootApplication
@Configuration
@RestController
public class ApplicationA extends WebMvcConfigurerAdapter {
public static void main(String[] args) {
SpringApplication.run(ApplicationA.class, args);
}
@RequestMapping("/test")
public Object test(HttpServletRequest request, HttpServletResponse response) {
Map<String, Object> map = new HashMap<>();
map.put("success", true);
map.put("msg", "我來(lái)自服務(wù)端");
return map;
}
// 跨域支持
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/**")
.allowedOrigins("*")
.allowCredentials(true)
.allowedMethods("GET", "POST", "DELETE", "PUT")
.maxAge(3600);
}Java中設(shè)置多個(gè)Access-Control-Allow-Origin跨域訪問
1、如果服務(wù)端是Java開發(fā)的,添加如下設(shè)置允許跨域即可,但是這樣做是允許所有域名都可以訪問,不夠安全。
response.setHeader(“Access-Control-Allow-Origin”,"*");
2、為保證安全性,可以只添加部分域名允許訪問,添加位置可以在下面三處任選一個(gè)。
(1)可以在過濾器的filter的dofilter()方法種設(shè)置。
(2)可以在servlet的get或者post方法里面設(shè)置。
(3)可以放在訪問的jsp頁(yè)面第一行。
3、在此用第一種方法,注意web.xml配置過濾器(filter)。
public void doFilter(ServletRequest req, ServletResponse res,FilterChain chain) throws IOException, ServletException {
// 將ServletResponse轉(zhuǎn)換為HttpServletResponse
HttpServletResponse httpResponse = (HttpServletResponse) res;
// 如果不是80端口,需要將端口加上,如果是集群,則用Nginx的地址,同理不是80端口要加上端口
String [] allowDomain= {"http://www.baidu.com","http://123.456.789.10","http://123.16.12.23:8080"};
Set allowedOrigins= new HashSet(Arrays.asList(allowDomain));
String originHeader=((HttpServletRequest) req).getHeader("Origin");
if (allowedOrigins.contains(originHeader)){
httpResponse.setHeader("Access-Control-Allow-Origin", originHeader);
httpResponse.setContentType("application/json;charset=UTF-8");
httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE");
httpResponse.setHeader("Access-Control-Max-Age", "3600");
httpResponse.setHeader("Access-Control-Allow-Headers", "Content-Type,Access-Token");
// 如果要把Cookie發(fā)到服務(wù)器,需要指定Access-Control-Allow-Credentials字段為true
httpResponse.setHeader("Access-Control-Allow-Credentials", "true");
httpResponse.setHeader("Access-Control-Expose-Headers", "*");
}
chain.doFilter(req, res);
}基于nginx配置請(qǐng)求的CORS
目前很多請(qǐng)求都不是直接暴露的,很多通過nginx做反向代理,因此可以使用nginx配置固定請(qǐng)求的Access-Control-Allow-Origin,實(shí)現(xiàn)跨域訪問。方式是在被請(qǐng)求的接口,配置location代理,添加header實(shí)現(xiàn)。
#活動(dòng)訪問接口跨域配置
location /promotion/activityPro {
proxy_pass http://frontHost/promotion/activityPro;
add_header 'Access-Control-Allow-Origin' '*';
#add_header 'Access-Control-Allow-Methods' 'GET, POST';
#add_header 'Access-Control-Allow-Credentials' "true";
#add_header 'Access-Control-Max-Age' 86400;
#add_header 'Access-Control-Allow-Header' 'Content-Type,*';
}三、前端跨域JSONP方案
有前端經(jīng)驗(yàn)的童鞋知道,有時(shí)我們會(huì)在自己的代碼里直接引入其它域名的js、css等靜態(tài)文件。為啥這些靜態(tài)文件沒被瀏覽器限制呢?通常為了減輕web服務(wù)器的壓力,我們會(huì)把js、css,img等靜態(tài)資源分離到另一臺(tái)獨(dú)立域名的服務(wù)器上,使其和前端分離開。基于這個(gè)原因,瀏覽器并沒有限制這類靜態(tài)資源的跨域訪問。
我們可以動(dòng)態(tài)地創(chuàng)建一個(gè)script,讓瀏覽器以為我們要獲取靜態(tài)資源,從而網(wǎng)開一面。而服務(wù)器端也需要做一點(diǎn)改變,不能直接返回json,而是返回一個(gè)立即執(zhí)行的函數(shù),而前端請(qǐng)求的結(jié)果就作為函數(shù)的參數(shù)。
后端接口返回
@SpringBootApplication
@Configuration
@RestController
public class ApplicationA {
public static void main(String[] args) {
SpringApplication.run(ApplicationA.class, args);
}
@RequestMapping("/test")
public String test(HttpServletRequest request, HttpServletResponse response, String callback)
throws IOException {
Map<String, Object> map = new HashMap<>();
map.put("success", true);
map.put("msg", "我來(lái)自服務(wù)端");
// 返回值如下:
// callback({"msg":"我來(lái)自服務(wù)端","success":true});
return String.format("%s(%s);", callback, JsonUtil.toJson(map));
}js原生實(shí)現(xiàn)jsonp
function test() {
// 外部域名,參數(shù)是和后端接口約定的callback指定接口返回后的回調(diào)函數(shù)
url = "http://localhost:8882/test?callback=_ajax_callback";
// 創(chuàng)建一個(gè)script元素
var script = document.createElement('script');
script.type = 'text/javascript';
script.src = url;
document.head.appendChild(script);
}
???????// 接口回調(diào)
function _ajax_callback(res) {
console.log("被回調(diào)了");
console.log(res);
}
jQuery實(shí)現(xiàn)jsonp
$.ajax({
url: 'http://localhost:8882/test',
type: 'get',
dataType: 'jsonp', // 請(qǐng)求方式
jsonpCallback: "_ajax_callback", // 回調(diào)函數(shù)名
data: {}
});
vue.js實(shí)現(xiàn)jsonp
this.$http.jsonp(‘http://localhost:8882/test', {undefined
params: {},
jsonp: ‘_ajax_callback'
}).then((res) => {undefined
console.log(res);
})JSONP的優(yōu)缺點(diǎn)
優(yōu)點(diǎn):它不像XMLHttpRequest對(duì)象實(shí)現(xiàn)的Ajax請(qǐng)求那樣受到同源策略的限制;它的兼容性更好,在更加古老的瀏覽器中都可以運(yùn)行,不需要XMLHttpRequest或ActiveX的支持;并且在請(qǐng)求完畢后可以通過調(diào)用callback的方式回傳結(jié)果。
缺點(diǎn):它只支持GET請(qǐng)求而不支持POST等其它類型的HTTP請(qǐng)求;它只支持跨域HTTP請(qǐng)求這種情況,不能解決不同域的兩個(gè)頁(yè)面之間如何進(jìn)行JavaScript調(diào)用的問題。
其它方式支持跨域
nginx反向代理:前端訪問相同域名,nginx再根據(jù)需要把請(qǐng)求轉(zhuǎn)發(fā)到外部域名;
后端代理:在后端接口里先請(qǐng)求外部資源(比如用HttpClient),然后把結(jié)果返回給前端,這樣就不是跨域了;
其它:借助iframe、postMessage等也可實(shí)現(xiàn)跨域。
document.domain,window.name,web sockets
更多關(guān)于瀏覽器跨域解決方案請(qǐng)查看下面的相關(guān)鏈接
相關(guān)文章
微信小程序左右滑動(dòng)切換頁(yè)面詳解及實(shí)例代碼
這篇文章主要介紹了微信小程序左右滑動(dòng)切換頁(yè)面詳解及實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02
微信小程序封裝http訪問網(wǎng)絡(luò)庫(kù)實(shí)例代碼
這篇文章主要介紹了微信小程序封裝http訪問網(wǎng)絡(luò)庫(kù)實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-05-05
微信小程序聯(lián)網(wǎng)請(qǐng)求的輪播圖
這篇文章主要介紹了微信小程序聯(lián)網(wǎng)請(qǐng)求的輪播圖的相關(guān)資料,需要的朋友可以參考下2017-07-07
Javascript使用integrity屬性進(jìn)行安全驗(yàn)證
這篇文章主要介紹了Javascript使用integrity屬性進(jìn)行安全驗(yàn)證,在html中,script標(biāo)簽可以通過src屬性引入一個(gè)js文件,引入的js文件可以是本地的,也可以是遠(yuǎn)程的,下面我們一起來(lái)看看文章詳細(xì)內(nèi)容2021-11-11
微信小程序中實(shí)現(xiàn)一對(duì)多發(fā)消息詳解及實(shí)例代碼
這篇文章主要介紹了微信小程序中實(shí)現(xiàn)一對(duì)多發(fā)消息詳解及實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02

