什麼是跨域?跨域的解決辦法

一、爲什麼會出現跨域問題

出於瀏覽器的同源策略限制。同源策略(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(此方案僅限主域相同,子域不同的跨域應用場景。)

  1. // 兩個頁面都設置
  2. document.domain = 'test.com';

【2】跨文檔通信 API:window.postMessage()

調用postMessage方法實現父窗口http://test1.com向子窗口http://test2.com發消息(子窗口同樣可以通過該方法發送消息給父窗口)

它可用於解決以下方面的問題:

  • 頁面和其打開的新窗口的數據傳遞
  • 多窗口之間消息傳遞
  • 頁面與嵌套的iframe消息傳遞
  • 上面三個場景的跨域數據傳遞
  1. // 父窗口打開一個子窗口
  2. var openWindow = window.open('http://test2.com', 'title');
  3. // 父窗口向子窗口發消息(第一個參數代表發送的內容,第二個參數代表接收消息窗口的url)
  4. openWindow.postMessage('Nice to meet you!', 'http://test2.com');

調用message事件,監聽對方發送的消息

  1. // 監聽 message 消息
  2. window.addEventListener('message', function (e) {
  3. console.log(e.source); // e.source 發送消息的窗口
  4. console.log(e.origin); // e.origin 消息發向的網址
  5. console.log(e.data); // e.data 發送的消息
  6. },false);

【3】JSONP

JSONP 是服務器與客戶端跨源通信的常用方法。最大特點就是簡單適用,兼容性好(兼容低版本IE),缺點是隻支持get請求,不支持post請求。

核心思想:網頁通過添加一個<script>元素,向服務器請求 JSON 數據,服務器收到請求後,將數據放在一個指定名字的回調函數的參數位置傳回來。

①原生實現:

  1. <script src="http://test.com/data.php?callback=dosomething"></script>
  2. // 向服務器test.com發出請求,該請求的查詢字符串有一個callback參數,用來指定回調函數的名字
  3. // 處理服務器返回回調函數的數據
  4. <script type="text/javascript">
  5. function dosomething(res){
  6. // 處理獲得的數據
  7. console.log(res.data)
  8. }
  9. </script>

② jQuery ajax:

  1. $.ajax({
  2. url: 'http://www.test.com:8080/login',
  3. type: 'get',
  4. dataType: 'jsonp', // 請求方式爲jsonp
  5. jsonpCallback: "handleCallback", // 自定義回調函數名
  6. data: {}
  7. });

③ Vue.js

  1. this.$http.jsonp('http://www.domain2.com:8080/login', {
  2. params: {},
  3. jsonp: 'handleCallback'
  4. }).then((res) => {
  5. console.log(res);
  6. })

【4】CORS

CORS 是跨域資源分享(Cross-Origin Resource Sharing)的縮寫。它是 W3C 標準,屬於跨源 AJAX 請求的根本解決方法。

1、普通跨域請求:只需服務器端設置Access-Control-Allow-Origin

2、帶cookie跨域請求:前後端都需要進行設置

【前端設置】根據xhr.withCredentials字段判斷是否帶有cookie

①原生ajax

  1. var xhr = new XMLHttpRequest(); // IE8/9需用window.XDomainRequest兼容
  2. // 前端設置是否帶cookie
  3. xhr.withCredentials = true;
  4. xhr.open('post', 'http://www.domain2.com:8080/login', true);
  5. xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
  6. xhr.send('user=admin');
  7. xhr.onreadystatechange = function() {
  8. if (xhr.readyState == 4 && xhr.status == 200) {
  9. alert(xhr.responseText);
  10. }
  11. };

② jQuery ajax 

  1. $.ajax({
  2. url: 'http://www.test.com:8080/login',
  3. type: 'get',
  4. data: {},
  5. xhrFields: {
  6. withCredentials: true // 前端設置是否帶cookie
  7. },
  8. crossDomain: true, // 會讓請求頭中包含跨域的額外信息,但不會含cookie
  9. });

③vue-resource

Vue.http.options.credentials = true

④ axios 

axios.defaults.withCredentials = true

【服務端設置】

服務器端對於CORS的支持,主要是通過設置Access-Control-Allow-Origin來進行的。如果瀏覽器檢測到相應的設置,就可以允許Ajax進行跨域的訪問。

① Java後臺

  1. /*
  2. * 導入包:import javax.servlet.http.HttpServletResponse;
  3. * 接口參數中定義:HttpServletResponse response
  4. */
  5. // 允許跨域訪問的域名:若有端口需寫全(協議+域名+端口),若沒有端口末尾不用加'/'
  6. response.setHeader("Access-Control-Allow-Origin", "http://www.domain1.com");
  7. // 允許前端帶認證cookie:啓用此項後,上面的域名不能爲'*',必須指定具體的域名,否則瀏覽器會提示
  8. response.setHeader("Access-Control-Allow-Credentials", "true");
  9. // 提示OPTIONS預檢時,後端需要設置的兩個常用自定義頭
  10. response.setHeader("Access-Control-Allow-Headers", "Content-Type,X-Requested-With");

② Nodejs後臺

  1. var http = require('http');
  2. var server = http.createServer();
  3. var qs = require('querystring');
  4. server.on('request', function(req, res) {
  5. var postData = '';
  6. // 數據塊接收中
  7. req.addListener('data', function(chunk) {
  8. postData += chunk;
  9. });
  10. // 數據接收完畢
  11. req.addListener('end', function() {
  12. postData = qs.parse(postData);
  13. // 跨域後臺設置
  14. res.writeHead(200, {
  15. 'Access-Control-Allow-Credentials': 'true', // 後端允許發送Cookie
  16. 'Access-Control-Allow-Origin': 'http://www.domain1.com', // 允許訪問的域(協議+域名+端口)
  17. /*
  18. * 此處設置的cookie還是domain2的而非domain1,因爲後端也不能跨域寫cookie(nginx反向代理可以實現),
  19. * 但只要domain2中寫入一次cookie認證,後面的跨域接口都能從domain2中獲取cookie,從而實現所有的接口都能跨域訪問
  20. */
  21. 'Set-Cookie': 'l=a123456;Path=/;Domain=www.domain2.com;HttpOnly' // HttpOnly的作用是讓js無法讀取cookie
  22. });
  23. res.write(JSON.stringify(postData));
  24. res.end();
  25. });
  26. });
  27. server.listen('8080');
  28. console.log('Server is running at port 8080...');

③ PHP後臺

  1. <?php
  2. 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="{&quot;mod&quot;:&quot;1585297308_001&quot;,&quot;dest&quot;:&quot;https://blog.csdn.net/qq_38128179/article/details/84956552&quot;,&quot;extend1&quot;:&quot;pc&quot;,&quot;ab&quot;:&quot;new&quot;}"><div></div></div>
                            </div>
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章