九种跨域方式的实现原理,第一个就超惊艳!( 三 )


  • 页面和其打开的新窗口的数据传递
  • 多窗口之间消息传递
  • 页面与嵌套的iframe消息传递
  • 上面三个场景的跨域数据传递
postMessage()方法允许来自不同源的脚本采用异步方式进行有限的通信,可以实现跨文本档、多窗口、跨域消息传递 。
otherWindow.postMessage(message, targetOrigin, [transfer]);
  • message: 将要发送到其他 window的数据 。
  • targetOrigin:通过窗口的origin属性来指定哪些窗口能接收到消息事件,其值可以是字符串"*"(表示无限制)或者一个URI 。在发送消息的时候,如果目标窗口的协议、主机地址或端口这三者的任意一项不匹配targetOrigin提供的值,那么消息就不会被发送;只有三者完全匹配,消息才会被发送 。
  • transfer(可选):是一串和message 同时传递的 Transferable 对象. 这些对象的所有权将被转移给消息的接收方,而发送一方将不再保有所有权 。
接下来我们看个例子: http://localhost:3000/a.html页面向 http://localhost:4000/b.html传递“我爱你”,然后后者传回"我不爱你" 。
// a.html <iframe src=https://www.isolves.com/it/cxkf/bk/2019-07-01/"http://localhost:4000/b.html" frameborder="0" id="frame" onload="load()"> //等它加载完触发一个事件 //内嵌在http://localhost:3000/a.html // b.html window.onmessage = function(e) { console.log(e.data) //我爱你 e.source.postMessage('我不爱你', e.origin)}4.websocket
Websocket是HTML5的一个持久化的协议,它实现了浏览器与服务器的全双工通信,同时也是跨域的一种解决方案 。WebSocket和HTTP都是应用层协议,都基于 TCP 协议 。
但是WebSocket是一种双向通信协议,在建立连接之后,WebSocket的server与 client都能主动向对方发送或接收数据 。
同时,WebSocket 在建立连接时需要借助 HTTP 协议,连接建立好了之后 client 与 server 之间的双向通信就与 HTTP 无关了 。
原生WebSocket API使用起来不太方便,我们使用 Socket.io,它很好地封装了webSocket接口,提供了更简单、灵活的接口,也对不支持webSocket的浏览器提供了向下兼容 。
我们先来看个例子:本地文件socket.html向 localhost:3000发生数据和接受数据 。
// socket.html<script> let socket = new WebSocket('ws://localhost:3000'); socket.onopen = function () { socket.send('我爱你');//向服务器发送数据 } socket.onmessage = function (e) { console.log(e.data);//接收服务器返回的数据 }</script>// server.jslet express = require('express');let app = express();let WebSocket = require('ws');//记得安装wslet wss = new WebSocket.Server({port:3000});wss.on('connection',function(ws) { ws.on('message', function (data) { console.log(data); ws.send('我不爱你') });})5. Node中间件代理(两次跨域)
实现原理:同源策略是浏览器需要遵循的标准,而如果是服务器向服务器请求就无需遵循同源策略 。代理服务器,需要做以下几个步骤:
  • 接受客户端请求。
  • 将请求 转发给服务器 。
  • 拿到服务器 响应 数据 。
  • 将 响应 转发给客户端 。

九种跨域方式的实现原理,第一个就超惊艳!

文章插图
 
我们先来看个例子:本地文件index.html文件,通过代理服务器 http://localhost:3000向目标服务器 http://localhost:4000请求数据 。
// index.html(http://127.0.0.1:5500)<script src="https://cdn.bootcss.com/jquery/3.3.1/jquery.min.js"></script> <script> $.ajax({ url: 'http://localhost:3000', type: 'post', data: { name: 'xiamen', password: '123456' }, contentType: 'application/json;charset=utf-8', success: function(result) { console.log(result) // {"title":"fontend","password":"123456"} }, error: function(msg) { console.log(msg) } }) </script>// server1.js 代理服务器(http://localhost:3000)const http = require('http')// 第一步:接受客户端请求const server = http.createServer((request, response) => { // 代理服务器,直接和浏览器直接交互,需要设置CORS 的首部字段 response.writeHead(200, { 'Access-Control-Allow-Origin': '*', 'Access-Control-Allow-Methods': '*', 'Access-Control-Allow-Headers': 'Content-Type' }) // 第二步:将请求转发给服务器 const proxyRequest = http .request( { host: '127.0.0.1', port: 4000, url: '/', method: request.method, headers: request.headers }, serverResponse => { // 第三步:收到服务器的响应 var body = '' serverResponse.on('data', chunk => { body += chunk }) serverResponse.on('end', () => { console.log('The data is ' + body) // 第四步:将响应结果转发给浏览器 response.end(body) }) } ) .end()})server.listen(3000, () => { console.log('The proxyServer is running at http://localhost:3000')})// server2.js(http://localhost:4000)const http = require('http')const data = { title: 'fontend', password: '123456' }const server = http.createServer((request, response) => { if (request.url === '/') { response.end(JSON.stringify(data)) }})server.listen(4000, () => { console.log('The server is running at http://localhost:4000')})


推荐阅读