出於瀏覽器的同源策略限制。同源策略(Sameoriginpolicy)是一種約定,它是瀏覽器最核心也最基本的安全功能,如果缺少了同源策略,則瀏覽器的正常功能可能都會受到影響。可以說Web是構建在同源策略基礎之上的,瀏覽器只是針對同源策略的一種實現。同源策略會阻止一個域的javascript腳本和另外一個域的內容進行交互。所謂同源(即指在同一個域)就是兩個頁面具有相同的協議(protocol),主機(host)和端口號(port)
二、什麼是跨域
當一個請求url的協議、域名、端口三者之間任意一個與當前頁面url不同即爲跨域
當前頁面url | 被請求頁面url | 是否跨域 | 原因 |
http://www.test.com/ | http://www.test.com/index.html | 否 | 同源(協議、域名、端口號相同) |
http://www.test.com/ | https://www.test.com/index.html | 跨域 | 協議不同(http/https) |
http://www.test.com/ | http://www.baidu.com/ | 跨域 | 主域名不同(test/baidu) |
http://www.test.com/ | http://blog.test.com/ | 跨域 | 子域名不同(www/blog) |
http://www.test.com:8080/ | http://www.test.com:7001/ | 跨域 | 端口號不同(8080/7001) |
三、非同源限制
【1】無法讀取非同源網頁的 Cookie、LocalStorage 和 IndexedDB
【2】無法接觸非同源網頁的 DOM
【3】無法向非同源地址發送 AJAX 請求
四、跨域解決方法
【1】設置document.domain解決無法讀取非同源網頁的 Cookie問題
因爲瀏覽器是通過document.domain屬性來檢查兩個頁面是否同源,因此只要通過設置相同的document.domain,兩個頁面就可以共享Cookie(此方案僅限主域相同,子域不同的跨域應用場景。)
- // 兩個頁面都設置
- document.domain = 'test.com';
【2】跨文檔通信 API:window.postMessage()
調用postMessage方法實現父窗口http://test1.com向子窗口http://test2.com發消息(子窗口同樣可以通過該方法發送消息給父窗口)
它可用於解決以下方面的問題:
- 頁面和其打開的新窗口的數據傳遞
- 多窗口之間消息傳遞
- 頁面與嵌套的iframe消息傳遞
- 上面三個場景的跨域數據傳遞
- // 父窗口打開一個子窗口
- var openWindow = window.open('http://test2.com', 'title');
-
- // 父窗口向子窗口發消息(第一個參數代表發送的內容,第二個參數代表接收消息窗口的url)
- openWindow.postMessage('Nice to meet you!', 'http://test2.com');
調用message事件,監聽對方發送的消息
- // 監聽 message 消息
- window.addEventListener('message', function (e) {
- console.log(e.source); // e.source 發送消息的窗口
- console.log(e.origin); // e.origin 消息發向的網址
- console.log(e.data); // e.data 發送的消息
- },false);
【3】JSONP
JSONP 是服務器與客戶端跨源通信的常用方法。最大特點就是簡單適用,兼容性好(兼容低版本IE),缺點是隻支持get請求,不支持post請求。
核心思想:網頁通過添加一個<script>元素
,向服務器請求 JSON 數據,服務器收到請求後,將數據放在一個指定名字的回調函數的參數位置傳回來。
①原生實現:
- <script src="http://test.com/data.php?callback=dosomething"></script>
- // 向服務器test.com發出請求,該請求的查詢字符串有一個callback參數,用來指定回調函數的名字
-
- // 處理服務器返回回調函數的數據
- <script type="text/javascript">
- function dosomething(res){
- // 處理獲得的數據
- console.log(res.data)
- }
- </script>
② jQuery ajax:
- $.ajax({
- url: 'http://www.test.com:8080/login',
- type: 'get',
- dataType: 'jsonp', // 請求方式爲jsonp
- jsonpCallback: "handleCallback", // 自定義回調函數名
- data: {}
- });
③ Vue.js
- this.$http.jsonp('http://www.domain2.com:8080/login', {
- params: {},
- jsonp: 'handleCallback'
- }).then((res) => {
- console.log(res);
- })
【4】CORS
CORS 是跨域資源分享(Cross-Origin Resource Sharing)的縮寫。它是 W3C 標準,屬於跨源 AJAX 請求的根本解決方法。
1、普通跨域請求:只需服務器端設置Access-Control-Allow-Origin
2、帶cookie跨域請求:前後端都需要進行設置
【前端設置】根據xhr.withCredentials字段判斷是否帶有cookie
①原生ajax
- var xhr = new XMLHttpRequest(); // 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);
- }
- };
② jQuery ajax
- $.ajax({
- url: 'http://www.test.com:8080/login',
- type: 'get',
- data: {},
- xhrFields: {
- withCredentials: true // 前端設置是否帶cookie
- },
- crossDomain: true, // 會讓請求頭中包含跨域的額外信息,但不會含cookie
- });
-
-
③vue-resource
Vue.http.options.credentials = true
④ axios
axios.defaults.withCredentials = true
【服務端設置】
服務器端對於CORS的支持,主要是通過設置Access-Control-Allow-Origin來進行的。如果瀏覽器檢測到相應的設置,就可以允許Ajax進行跨域的訪問。
① Java後臺
- /*
- * 導入包:import javax.servlet.http.HttpServletResponse;
- * 接口參數中定義:HttpServletResponse response
- */
-
- // 允許跨域訪問的域名:若有端口需寫全(協議+域名+端口),若沒有端口末尾不用加'/'
- response.setHeader("Access-Control-Allow-Origin", "http://www.domain1.com");
-
- // 允許前端帶認證cookie:啓用此項後,上面的域名不能爲'*',必須指定具體的域名,否則瀏覽器會提示
- response.setHeader("Access-Control-Allow-Credentials", "true");
-
- // 提示OPTIONS預檢時,後端需要設置的兩個常用自定義頭
- response.setHeader("Access-Control-Allow-Headers", "Content-Type,X-Requested-With");
② Nodejs後臺
- var http = require('http');
- var server = http.createServer();
- var qs = require('querystring');
-
- server.on('request', function(req, res) {
- var postData = '';
-
- // 數據塊接收中
- req.addListener('data', function(chunk) {
- postData += chunk;
- });
-
- // 數據接收完畢
- req.addListener('end', function() {
- postData = qs.parse(postData);
-
- // 跨域後臺設置
- res.writeHead(200, {
- 'Access-Control-Allow-Credentials': 'true', // 後端允許發送Cookie
- 'Access-Control-Allow-Origin': 'http://www.domain1.com', // 允許訪問的域(協議+域名+端口)
- /*
- * 此處設置的cookie還是domain2的而非domain1,因爲後端也不能跨域寫cookie(nginx反向代理可以實現),
- * 但只要domain2中寫入一次cookie認證,後面的跨域接口都能從domain2中獲取cookie,從而實現所有的接口都能跨域訪問
- */
- 'Set-Cookie': 'l=a123456;Path=/;Domain=www.domain2.com;HttpOnly' // HttpOnly的作用是讓js無法讀取cookie
- });
-
- res.write(JSON.stringify(postData));
- res.end();
- });
- });
-
- server.listen('8080');
- console.log('Server is running at port 8080...');
③ PHP後臺
- <?php
- header("Access-Control-Allow-Origin:*");
④ Apache需要使用mod_headers模塊來激活HTTP頭的設置,它默認是激活的。你只需要在Apache配置文件的<Directory>, <Location>, <Files>或<VirtualHost>的配置里加入以下內容即可
Header set Access-Control-Allow-Origin *
文章每週持續更新,可以微信搜索「 前端大集錦 」第一時間閱讀,回覆【視頻】【書籍】領取200G視頻資料和30本PDF書籍資料
</div><div data-report-view="{"mod":"1585297308_001","dest":"https://blog.csdn.net/qq_38128179/article/details/84956552","extend1":"pc","ab":"new"}"><div></div></div>
</div>