网站首页 > 博客文章 正文
将跨页面通讯类比计算机进程间的通讯,其实方法无外乎那么几种,而web领域可以实现的技术方案主要是类似于以下两种原理:
- 获取句柄,定向通讯
- 共享内存,结合轮询或者事件通知来完成业务逻辑
由于第二种原理更利于解耦业务逻辑,具体的实现方案比较多样。以下是具体的实现方案,简单介绍下,权当科普:
一、获取句柄
具体方案
父页面通过window.open(url, name)方式打开的子页面可以获取句柄,然后通过postMessage完成通讯需求。
// parent.html const childPage = window.open('child.html', 'child') childPage.onload = () => { childPage.postMessage('hello', location.origin) } // child.html window.onmessage = evt => { // evt.data }
tips
- 当指定window.open的第二个name参数时,再次调用window.open('****', 'child')会使之前已经打开的同name子页面刷新
- 由于安全策略,异步请求之后再调用window.open会被浏览器阻止,不过可以通过句柄设置子页面的url即可实现类似效果
// 首先先开一个空白页 const tab = window.open('about:blank') // 请求完成之后设置空白页的url fetch(/* ajax */).then(() => { tab.location.href = '****' })
优劣
缺点是只能与自己打开的页面完成通讯,应用面相对较窄;但优点是在跨域场景中依然可以使用该方案。
二、localStorage
具体方案
设置共享区域的storage,storage会触发storage事件
// A.html localStorage.setItem('message', 'hello') // B.html window.onstorage = evt => { // evt.key, evt.oldValue, evt.newValue }
tips
- 触发写入操作的页面下的storage listener不会被触发
- storage事件只有在发生改变的时候才会触发,即重复设置相同值不会触发listener
- safari隐身模式下无法设置localStorage值
优劣
API简单直观,兼容性好,除了跨域场景下需要配合其他方案,无其他缺点
三、BroadcastChannel
具体方案
和localStorage方案基本一致,额外需要初始化
// A.html const channel = new BroadcastChannel('tabs') channel.onmessage = evt => { // evt.data } // B.html const channel = new BroadcastChannel('tabs') channel.postMessage('hello')
优劣
和localStorage方案没特别区别,都是同域、API简单,BroadcastChannel方案兼容性差些(chrome > 58),但比localStorage方案生命周期短(不会持久化),相对干净些。
四、SharedWorker
具体方案
SharedWorker本身并不是为了解决通讯需求的,它的设计初衷应该是类似总控,将一些通用逻辑放在SharedWorker中处理。不过因为也能实现通讯,所以一并写下:
// A.html var sharedworker = new SharedWorker('worker.js') sharedworker.port.start() sharedworker.port.onmessage = evt => { // evt.data } // B.html var sharedworker = new SharedWorker('worker.js') sharedworker.port.start() sharedworker.port.postMessage('hello') // worker.js const 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了。
communication between browser windows (and tabs too) using cookies
优劣
相较于其他方案没有存在优势的地方,只能同域使用,而且污染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.io、sockjs例子比较多,略
消息队列
项目规模大型时,需要消息队列集群长时间维护长链接,在需要的时候进行广播。
提供该类服务的云服务商很多,或者寻找一些开源方案自建。
例如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接口调用频道广播接口
希望本文能帮助到您!
点赞+转发,让更多的人也能看到这篇内容(收藏不点赞,都是耍流氓-_-)
关注 {我},享受文章首发体验!
每周重点攻克一个前端技术难点。更多精彩前端内容私信 我 回复“教程”
原文链接:https://github.com/ProtoTeam/blog/blob/master/201709/3.md
作者:蚂蚁金服数据体验技术团队
猜你喜欢
- 2024-09-17 Node.js 22 发布(node.js还火吗)
- 2024-09-17 京东到家Loki日志系统实践(京东到家早起打卡入口)
- 2024-09-17 从 Element UI 源码的构建流程来看前端 UI 库设计
- 2024-09-17 第一届技术博文征文活动开启(庞博文图片)
- 2024-09-17 七天速记前端八股文(重点)(前端八股文文档)
- 2024-09-17 基于Swoole的高性能系统监控及Nginx负载均衡的实现
- 2024-09-17 Nodejs异步Generator函数和Websockets
- 2024-09-17 我用这11招,让接口性能提升了100倍
- 2024-09-17 超实用!程序员必备工具软件清单,来自腾讯工程师们的分享
- 2024-09-17 让程序员头疼的微服务下数据聚合join(二)
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- powershellfor (55)
- messagesource (56)
- aspose.pdf破解版 (56)
- promise.race (63)
- 2019cad序列号和密钥激活码 (62)
- window.performance (66)
- qt删除文件夹 (72)
- mysqlcaching_sha2_password (64)
- ubuntu升级gcc (58)
- nacos启动失败 (64)
- ssh-add (70)
- jwt漏洞 (58)
- macos14下载 (58)
- yarnnode (62)
- abstractqueuedsynchronizer (64)
- source~/.bashrc没有那个文件或目录 (65)
- springboot整合activiti工作流 (70)
- jmeter插件下载 (61)
- 抓包分析 (60)
- idea创建mavenweb项目 (65)
- vue回到顶部 (57)
- qcombobox样式表 (68)
- vue数组concat (56)
- tomcatundertow (58)
- pastemac (61)
本文暂时没有评论,来添加一个吧(●'◡'●)