同源策略

之所以会出现跨域问题,其实是由于浏览器同源策略的存在。同源策略是浏览器最基本最核心的功能,是一种安全策略。同源策略是浏览器行为,是为了保护本地数据不被获取回来的数据污染,因此拦截时机是在服务端返回时,即请求已经发出,服务器已经处理,只是在响应返回时,浏览器因为请求跨域而拒绝处理此次回来的数据。

跨域

说到跨域,我们先了解一下同域(同源),我们在控制台打印一下window.location.origin,可以看到origin由三个部分组成:origin = protocol+hostname+port,所以同域代表protocol、hostname、port都要相同,那么跨域就是这三个只要其中一个不同就带表跨域。以下面绿色行对比举例:

protocol hostname port 是否跨域
http www.a.com 80
https www.a.com 80 Y
http yyy.a.comm 80 Y
http www.a.com 90 Y

image.png

如何解决

1、JSONP

  1. <script>
  2. var script = document.createElement('script');
  3. script.type = 'text/javascript';
  4. // 传参一个回调函数名给后端,方便后端返回时执行这个在前端定义的回调函数
  5. script.src = 'http://www.domain.com/path?callback=handleCallback';
  6. document.head.appendChild(script);
  7. // 回调执行函数
  8. function handleCallback(res) {
  9. alert(JSON.stringify(res));
  10. }
  11. </script>

2、document.domain + iframe

此方案仅限主域相同,子域不同的情况,设置两个页面的document.domain = 主域,举例:
父窗口:http://parent.domain.com/page.html

  1. <iframe id="iframe" src="http://child.domain.com/page.html"></iframe>
  2. <script>
  3. document.domain = "domain.com"
  4. const parentName = "wstreet7"
  5. const childWindow = document.getElementById('iframe').contentWindow
  6. const { childName } = childWindow
  7. </script>

子窗口:http://child.domain.com/page.html

  1. <script>
  2. document.domain = "domain.com"
  3. const { parentName } = window.parent
  4. console.log(parentName) // wstreet7
  5. const childName = 'child'
  6. </script>

3、location.hash + iframe

页面通信一般是双向通信,A页面可以获取B页面的数据,B页面也可以获取A页面的数据使用location.hash + iframe达到两个页面通信是需要一个代理C页面的
A页面:www.domain1.com/a.html

  1. <iframe id="iframe-b" src="http://www.domain2.com/b.html" style="display:none;"></iframe>
  2. <script>
  3. var iframe = document.getElementById('iframe-b');
  4. // 向B.html传hash值
  5. setTimeout(function() {
  6. iframe.src = iframe.src + '#user=wstreet';
  7. }, 1000);
  8. // 开放给同域c.html的回调方法
  9. function onCallback(res) {
  10. alert('data from c.html ---> ' + res);
  11. }
  12. </script>

B页面:www.domain2.com/b.html

  1. <iframe id="iframe-c" src="http://www.domain1.com/c.html" style="display:none;"></iframe>
  2. <script>
  3. var iframe = document.getElementById('iframe-c');
  4. // 监听a.html传来的hash值,再传给c.html
  5. window.onhashchange = function () {
  6. const user = location.hash.replace('#user=', '');
  7. // 向C页面传值,C页面再传回给A页面
  8. iframe.src = iframe.src + '#cAge=26'
  9. };
  10. </script>

C页面:http://www.domain1.com/c.html

  1. <script>
  2. // 监听b.html传来的hash值
  3. window.onhashchange = function () {
  4. // 再通过操作同域a.html的js回调,将结果传回
  5. window.parent.parent.onCallback('age: ' + location.hash.replace('#cAge=', ''));
  6. };
  7. </script>

4、window.name + iframe

此方法利用了window.name在location改变之后数据依然不会改变的特性来实现跨域
A页面:http//domain1.com/a.html

  1. const proxy = (url, callback) => {
  2. let state = 0
  3. const iframe = document.createElement('iframe')
  4. // 会调用2次onload
  5. iframe.src = url
  6. iframe.onload = () => {
  7. if (state === 0) {
  8. // 第1次,加载跨域b页面成功后,设置location,切到同域代理页面
  9. // 只有同域情况下,才能获取iframe.contentWindow
  10. iframe.contentWindow.location = 'http://www.domain1.com/proxy.html'
  11. state = 1
  12. } else if (state === 1) {
  13. // 第2次加载同域代理页面成功后读取iframe.name
  14. callback(iframe.contentWindow.name)
  15. // 获取到数据后销毁iframe
  16. destoryFrame()
  17. }
  18. }
  19. document.body.appendChild(iframe);
  20. function destoryFrame() {
  21. iframe.contentWindow.document.write('');
  22. iframe.contentWindow.close();
  23. document.body.removeChild(iframe);
  24. }
  25. }
  26. proxy('http://www.domain2.b.html', (data) => {
  27. console.log(data)
  28. })

B页面:http://www.domain2.com/b.html

  1. <script>
  2. window.name = 'wstreet'
  3. </script>

proxy页面:http://www.domain1.com/proxy.html(空白页面)

总结: 1、iframe先加载跨域页面及,将需要获取的数据保存在跨域页面的windo.name; 2、改变iframe的地址到同域代理页面,这时候可以读取iframe.contentWindow

5、postMessage

postMessage方法允许来自不同源的脚本采用异步方式进行有限的通信,可以实现跨文本档、多窗口、跨域消息传递,postMessage可以解决以下问题:
1)、页面和其他打开的新窗口的数据传递
2)、多窗口之间消息传递
3)、页面和嵌套的iframe之间的消息传递
4)、以上三个场景跨域数据传递
用法:postMessage(data,origin)接受两个参数:
(1)data:HTML5规范中规定该参数可以是JavaScript任意基本类型数据或者可复制的对象。然而并不是所有浏览器都支持,有些浏览器只支持字符串,所以我们在传递参数的时候需要使用JSON.stringify()方法对对象参数序列化。
(2)origin:字符串类型,指明目标窗口的源,协议+主机+端口号[+URL],URL会被忽略,所以可以不写,这个参数是为了安全考虑,postMessage只会将message传给指定窗口。也可以将origin设置成“*”,这样可以传递给任意窗口。如果要指定接收的窗口要和当前窗口同源的话,可以将origin设置成“/”。

A页面:http://www.domain1.com/a.html

  1. <iframe id="iframe" src="http://www.domain2.com/b.html" frameborder="0"></iframe>
  2. <script>
  3. const iframe = document.querySelector('#iframe')
  4. iframe.onload = () => {
  5. iframe.contentWindow.postMessage('wstreet', 'http://www.domain2.com')
  6. }
  7. window.addEventListener('message', e => {
  8. console.log(e.data)
  9. })
  10. </script>

B页面:http://www.domain2.com/b.html

  1. <script>
  2. window.addEventListener('message', e => {
  3. const data = e.data
  4. window.parent.postMessage(data, 'http://www.domain1.com')
  5. })
  6. </script>

6、CORS(跨域资源共享)

需要服务端设置一些请求头

  1. // 允许跨域访问的域名
  2. response.setHeader("Access-Control-Allow-Origin", "http://www.domain1.com");
  3. // 允许前端带认证cookie,若此项开启,上面的Access-Control-Allow-Origin不能设置成"*"
  4. response.setHeader("Access-Control-Allow-Credentials", "true");
  5. // options预检时,后端需要设置的两个常用自定义头
  6. response.setHeader("Access-Control-Allow-Headers", "Content-Type,X-Requested-With");

如果需要携带cookie,前端需要设置

  1. xhr.withCredentials = true;

7、Nginx代理跨域

同源策略只是浏览器的安全策略,对HTTP不造成限制,当请求发出时,可以在中间使用Nginx做一下跳板,将相关内容做一下替换。

  1. server {
  2. listen 81;
  3. server_name www.domain1.com;
  4. location / {
  5. proxy_pass http://www.domain2.com:8080; #反向代理
  6. proxy_cookie_domain www.domain2.com www.domain1.com; #修改cookie里域名
  7. index index.html index.htm;
  8. # 当用webpack-dev-server等中间件代理接口访问nignx时,此时无浏览器参与,故没有同源限制,下面的跨域配置可不启用
  9. add_header Access-Control-Allow-Origin http://www.domain1.com; #当前端只跨域不带cookie时,可为*
  10. add_header Access-Control-Allow-Credentials true;
  11. }
  12. }

8、WebSocket协议跨域

WebSocket协议是HTML5一种新的协议,它实现了浏览器与服务器双工通信,同时允许跨域通信。
使用Socket.io演示:

(1)前端代码:

  1. //
  2. <div>user input:<input type="text"></div>
  3. <script src="https://cdn.bootcss.com/socket.io/2.2.0/socket.io.js"></script>
  4. <script>
  5. var socket = io('http://www.domain2.com:8080');
  6. // 连接成功处理
  7. socket.on('connect', function() {
  8. // 监听服务端消息
  9. socket.on('message', function(msg) {
  10. console.log('data from server: ---> ' + msg);
  11. });
  12. // 监听服务端关闭
  13. socket.on('disconnect', function() {
  14. console.log('Server socket has closed.');
  15. });
  16. });
  17. document.getElementsByTagName('input')[0].onblur = function() {
  18. socket.send(this.value);
  19. };
  20. </script>

(2)Nodejs socket后台

  1. var http = require('http');
  2. var socket = require('socket.io');
  3. // 启http服务
  4. var server = http.createServer(function(req, res) {
  5. res.writeHead(200, {
  6. 'Content-type': 'text/html'
  7. });
  8. res.end();
  9. });
  10. server.listen('8080');
  11. console.log('Server is running at port 8080...');
  12. // 监听socket连接
  13. socket.listen(server).on('connection', function(client) {
  14. // 接收信息
  15. client.on('message', function(msg) {
  16. client.send('hello:' + msg);
  17. console.log('data from client: ---> ' + msg);
  18. });
  19. // 断开处理
  20. client.on('disconnect', function() {
  21. console.log('Client socket has closed.');
  22. });
  23. });