博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
跨页面通信的各种姿势
阅读量:7237 次
发布时间:2019-06-29

本文共 3660 字,大约阅读时间需要 12 分钟。

作者简介:nekron 蚂蚁金服·数据体验技术团队

将跨页面通讯类比计算机进程间的通讯,其实方法无外乎那么几种,而web领域可以实现的技术方案主要是类似于以下两种原理:

  • 获取句柄,定向通讯
  • 共享内存,结合轮询或者事件通知来完成业务逻辑

由于第二种原理更利于解耦业务逻辑,具体的实现方案比较多样。以下是具体的实现方案,简单介绍下,权当科普:

一、获取句柄

具体方案

父页面通过window.open(url, name)方式打开的子页面可以获取句柄,然后通过postMessage完成通讯需求。

// parent.htmlconst childPage = window.open('child.html', 'child')childPage.onload = () => {	childPage.postMessage('hello', location.origin)}// child.htmlwindow.onmessage = evt => {	// evt.data}复制代码

tips

  1. 当指定window.open的第二个name参数时,再次调用window.open('****', 'child')会使之前已经打开的同name子页面刷新
  2. 由于安全策略,异步请求之后再调用window.open会被浏览器阻止,不过可以通过句柄设置子页面的url即可实现类似效果
// 首先先开一个空白页const tab = window.open('about:blank')// 请求完成之后设置空白页的urlfetch(/* ajax */).then(() => {	tab.location.href = '****'})复制代码

优劣

缺点是只能与自己打开的页面完成通讯,应用面相对较窄;但优点是在跨域场景中依然可以使用该方案。

二、localStorage

具体方案

设置共享区域的storage,storage会触发storage事件

// A.htmllocalStorage.setItem('message', 'hello')// B.htmlwindow.onstorage = evt => {  // evt.key, evt.oldValue, evt.newValue}复制代码

tips

  1. 触发写入操作的页面下的storage listener不会被触发
  2. storage事件只有在发生改变的时候才会触发,即重复设置相同值不会触发listener
  3. safari隐身模式下无法设置localStorage值

优劣

API简单直观,兼容性好,除了跨域场景下需要配合其他方案,无其他缺点

三、BroadcastChannel

具体方案

localStorage方案基本一致,额外需要初始化

// A.htmlconst channel = new BroadcastChannel('tabs')channel.onmessage = evt => {	// evt.data}// B.htmlconst channel = new BroadcastChannel('tabs')channel.postMessage('hello')复制代码

优劣

localStorage方案没特别区别,都是同域、API简单,BroadcastChannel方案兼容性差些(chrome > 58),但比localStorage方案生命周期短(不会持久化),相对干净些。

四、SharedWorker

具体方案

SharedWorker本身并不是为了解决通讯需求的,它的设计初衷应该是类似总控,将一些通用逻辑放在SharedWorker中处理。不过因为也能实现通讯,所以一并写下:

// A.htmlvar sharedworker = new SharedWorker('worker.js')sharedworker.port.start()sharedworker.port.onmessage = evt => {	// evt.data}// B.htmlvar sharedworker = new SharedWorker('worker.js')sharedworker.port.start()sharedworker.port.postMessage('hello')// worker.jsconst ports = []onconnect = e => {	const port = e.ports[0]	ports.push(port)	port.onmessage = evt => {		ports.filter(v => v!== port) // 此处为了贴近其他方案的实现,剔除自己		.forEach(p => p.postMessage(evt.data))	}}复制代码

优劣

相较于其他方案没有优势,此外,API复杂而且调试不方便。

五、Cookie

具体方案

一个古老的方案,有点localStorage的降级兼容版,我也是整理本文的时候才发现的,思路就是往document.cookie写入值,由于cookie的改变没有事件通知,所以只能采取轮询脏检查来实现业务逻辑。

方案比较丑陋,势必被淘汰的方案,贴一下原版思路地址,我就不写demo了。

优劣

相较于其他方案没有存在优势的地方,只能同域使用,而且污染cookie以后还额外增加AJAX的请求头内容。

六、Server

之前的方案都是前端自行实现,势必受到浏览器限制,比如无法做到跨浏览器的消息通讯,比如大部分方案都无法实现跨域通讯(需要增加额外的postMessage逻辑才能实现)。通过借助服务端,还有很多增强方案,也一并说下。

乞丐版

后端无开发量,前端定期保存,在tab被激活时重新获取保存的数据,可以通过校验hash之类的标记位来提升检查性能。

window.onvisibilitychange = () => {	if (document.visibilityState === 'visible') {		// AJAX	}}复制代码

Server-sent Events / Websocket

项目规模小型的时候可以采取这类方案,后端自行维护连接,以及后续的推送行为。

SSE

// 前端const es = new EventSource('/notification')es.onmessage = evt => {	// evt.data}es.addEventListener('close', () => {	es.close()}, false)// 后端,express为例const clients = []app.get('/notification', (req, res) => {	res.setHeader('Content-Type', 'text/event-stream')	clients.push(res)  	req.on('aborted', () => {   		// 清理clients  	})})app.get('/update', (req, res) => {	// 广播客户端新的数据	clients.forEach(client => {		client.write('data:hello\n\n')		setTimeout(() => {			client.write('event:close\ndata:close\n\n')		}, 500)	})	res.status(200).end()})复制代码

Websocket

socket.iosockjs例子比较多,略

消息队列

项目规模大型时,需要消息队列集群长时间维护长链接,在需要的时候进行广播。

提供该类服务的云服务商很多,或者寻找一些开源方案自建。

例如MQTT协议方案(阿里云就有提供),web客户端本质上也是websocket,需要集群同时支持ws和mqtt协议,示例如下:

// 前端// 客户端使用开源的Paho// port会和mqtt协议通道不同const client = new Paho.MQTT.Client(host, port, 'clientId')client.onMessageArrived = message => {	// message. payloadString}client.connect({	onSuccess: () => {		client.subscribe('notification')	}})// 抑或,借助flash(虽然快要被淘汰了)进行mqtt协议连接并订阅相应的频道,flash再通过回调抛出消息// 后端// 根据服务商提供的Api接口调用频道广播接口复制代码

原文地址:

转载地址:http://rigfm.baihongyu.com/

你可能感兴趣的文章
2015 年度 Android 开发者必备的 5 个开源库
查看>>
GLUT Trackball Demo
查看>>
SQL集合操作符样例(UNION,INTERSECT,EXCEPT)
查看>>
Spark-SparkSQL深入学习系列十一(转自OopsOutOfMemory)
查看>>
JAVA Eclipse使用Maven构建web项目详解(SSM框架)
查看>>
高校银行协同 信息校园一卡通行
查看>>
论MongoDB索引选择的重要性
查看>>
建设互联网医疗的思考
查看>>
澄清云计算概念 解析云计算现状
查看>>
应对5G网络需求,G.metro技术逐步走向成熟和应用
查看>>
【LeetCode从零单排】No104 Maximum Depth of Binary Tree
查看>>
使用使用Zopfli优化PNG图片
查看>>
德扑 AI 之父托马斯·桑德霍姆:扑克 AI 如何完虐人类,和 AlphaGo 大不同
查看>>
最常用的四种大数据分析方法
查看>>
极进网络石奇海:向802.11ac技术智能迁移的八大考虑要素
查看>>
中央财经大学创新创业中心主任尚超:大数据技术在防范虚假发票中的应用
查看>>
测试硬件,不妨试试这5款应用
查看>>
《数字逻辑设计与计算机组成》一3.8 实数算术
查看>>
把恶意程序存储到DNA上?黑客们的又一新发现
查看>>
ARM Tech Day:加速释放ARM IP,代号DynamIQ向人工智能进击
查看>>