JavaScript中實(shí)現(xiàn)跨標(biāo)簽頁通信的方法詳解
1.Broadcast Channel API(主流推薦)
Broadcast Channel API 允許不同標(biāo)簽頁之間進(jìn)行實(shí)時(shí)通信,而無需使用定時(shí)器輪詢。你可以創(chuàng)建一個(gè)共享的 Broadcast Channel,并在標(biāo)簽頁之間發(fā)送消息。這樣,當(dāng)其他標(biāo)簽頁接收到消息時(shí),可以立即作出響應(yīng)。Broadcast Channel API 提供了更可靠和高效的跨標(biāo)簽頁通信方式,避免了不必要的輪詢和性能開銷。
首先我們先創(chuàng)建一個(gè)文件crossTagMsg
const channel = new BroadcastChannel('sync-update') export function sendMsg(type: any, content: any) { channel.postMessage({ type, content }); } export function listenMsg(callback: (arg0: any) => any) { const handler = (e: { data: any; }) => { callback&&callback(e.data); } channel.addEventListener('message',handler) return ()=>{ channel.removeEventListener('message', handler) } }
很簡單的兩行代碼,一個(gè)發(fā)送,一個(gè)監(jiān)聽,監(jiān)聽返回一個(gè)移除函數(shù)
在頁面中引用函數(shù)就可以了
import {listenMsg,sendMsg} from '@/utils/crossTagMsg' const send = ()=>{ sendMsg('send',{name:'send'}) } const unmountedListenMsg = listenMsg((info:any)=>{ console.log(info.content) }) onUnmounted(unmountedListenMsg)
2.window.postMessage()
window.postMessage()
方法允許在不同的標(biāo)簽頁之間進(jìn)行跨域通信,并且是一種安全的方式來實(shí)現(xiàn)跨標(biāo)簽頁通信。下面是使用 window.postMessage()
進(jìn)行跨標(biāo)簽頁通信的基本步驟:
在發(fā)送消息的標(biāo)簽頁(發(fā)送方):
// 發(fā)送消息到目標(biāo)標(biāo)簽頁 const targetWindow = window.open('目標(biāo)標(biāo)簽頁的URL', '_blank'); // 打開目標(biāo)標(biāo)簽頁 targetWindow.postMessage('Hello from sender', '目標(biāo)標(biāo)簽頁的源'); // 監(jiān)聽來自目標(biāo)標(biāo)簽頁的響應(yīng)消息 window.addEventListener('message', function(event) { if (event.source === targetWindow) { console.log('Received response:', event.data); } });
在接收消息的標(biāo)簽頁(接收方):
// 監(jiān)聽來自發(fā)送方標(biāo)簽頁的消息 window.addEventListener('message', function(event) { if (event.origin === '發(fā)送方標(biāo)簽頁的源') { console.log('Received message:', event.data); // 發(fā)送響應(yīng)消息給發(fā)送方標(biāo)簽頁 event.source.postMessage('Hello from receiver', event.origin); } });
上述代碼中,發(fā)送方標(biāo)簽頁使用 window.open()
方法打開目標(biāo)標(biāo)簽頁,并使用 targetWindow.postMessage()
發(fā)送消息到目標(biāo)標(biāo)簽頁。在接收方標(biāo)簽頁中,使用 window.addEventListener()
監(jiān)聽來自發(fā)送方標(biāo)簽頁的消息,并在收到消息后發(fā)送響應(yīng)消息。
在 postMessage()
方法中,第一個(gè)參數(shù)是要發(fā)送的數(shù)據(jù),可以是字符串、數(shù)字、對象等。第二個(gè)參數(shù)是目標(biāo)窗口的源(origin),用于指定接收方標(biāo)簽頁的源。這是為了確保只有來自指定源的消息才會(huì)被接收和處理,以提供安全性。
需要注意的是,為了確保安全性,需要在通信的兩個(gè)標(biāo)簽頁之間建立信任關(guān)系,即確保目標(biāo)標(biāo)簽頁的源是你所期望的,并且不接受來自未知或不信任的源的消息。
通過使用 window.postMessage()
方法,你可以在不同的標(biāo)簽頁之間進(jìn)行跨域通信,并傳遞數(shù)據(jù)和消息。這種方式適用于需要在不同標(biāo)簽頁之間進(jìn)行實(shí)時(shí)通信或共享數(shù)據(jù)的場景。
3.visibilitychange
visibilitychange可以監(jiān)聽頁面的狀態(tài),判斷頁面的顯示和隱藏,當(dāng)頁面顯示的時(shí)候可以重新請求最新的數(shù)據(jù)
window.addEventListener('visibilitychange',function(e:any){ if (document.visibilityState === 'visible') { // 頁面顯示 console.log('頁面顯示'); // 執(zhí)行頁面顯示時(shí)的操作 更新數(shù)據(jù) } else if (document.visibilityState === 'hidden') { // 頁面隱藏 console.log('頁面隱藏'); // 執(zhí)行頁面隱藏時(shí)的操作 } })
4.Server-Sent Events(SSE)
Server-Sent Events(SSE)是一種用于在客戶端和服務(wù)器之間實(shí)現(xiàn)實(shí)時(shí)單向通信的技術(shù)。它允許服務(wù)器向客戶端推送數(shù)據(jù),而無需客戶端發(fā)送請求。
SSE 基于 HTTP 協(xié)議,使用了長輪詢(Long Polling)的機(jī)制??蛻舳送ㄟ^向服務(wù)器發(fā)送一個(gè) HTTP 請求,該請求保持打開狀態(tài),直到服務(wù)器有新的數(shù)據(jù)可用或連接超時(shí)。當(dāng)服務(wù)器有新數(shù)據(jù)時(shí),它會(huì)將數(shù)據(jù)作為一個(gè)持久連接的響應(yīng)發(fā)送給客戶端。
使用 SSE 的關(guān)鍵是服務(wù)器端需要發(fā)送特殊格式的數(shù)據(jù),遵循 SSE 的規(guī)范。服務(wù)器發(fā)送的數(shù)據(jù)被組織為一系列的事件(events),每個(gè)事件具有一個(gè)事件類型和數(shù)據(jù)字段??蛻舳送ㄟ^監(jiān)聽相應(yīng)的事件類型來處理接收到的數(shù)據(jù)。
下面是一個(gè)使用 SSE 的示例:
服務(wù)器端代碼(Node.js):
const http = require('http'); http.createServer((req, res) => { res.writeHead(200, { 'Content-Type': 'text/event-stream', 'Cache-Control': 'no-cache', 'Connection': 'keep-alive', }); // 發(fā)送事件及數(shù)據(jù)到客戶端 res.write('event: message\n'); res.write('data: Hello, world!\n\n'); // 模擬每秒鐘發(fā)送一個(gè)事件 setInterval(() => { res.write('event: message\n'); res.write(`data: Current time is ${new Date().toLocaleTimeString()}\n\n`); }, 1000); }).listen(3000);
客戶端代碼(JavaScript):
const eventSource = new EventSource('/stream'); eventSource.addEventListener('message', (event) => { const data = event.data; console.log('Received message:', data); });
在上述示例中,服務(wù)器創(chuàng)建了一個(gè) HTTP 服務(wù)器,當(dāng)客戶端向 /stream
路徑發(fā)出請求時(shí),服務(wù)器會(huì)將響應(yīng)的 Content-Type
設(shè)置為 text/event-stream
,表示這是一個(gè) SSE 連接。服務(wù)器通過發(fā)送以 "event" 和 "data" 字段組成的事件響應(yīng),向客戶端推送數(shù)據(jù)。
客戶端使用 EventSource
對象來建立 SSE 連接,并通過事件監(jiān)聽器來處理接收到的事件。在上述示例中,客戶端監(jiān)聽了 message
事件,并在事件發(fā)生時(shí)將數(shù)據(jù)打印到控制臺(tái)。
SSE 提供了一種簡單而有效的方式來實(shí)現(xiàn)服務(wù)器向客戶端的實(shí)時(shí)數(shù)據(jù)推送,適用于需要實(shí)時(shí)更新數(shù)據(jù)的應(yīng)用程序,如聊天應(yīng)用、實(shí)時(shí)通知、股票報(bào)價(jià)等。它相對于傳統(tǒng)的輪詢方式具有更低的延遲和更高的效率。
到此這篇關(guān)于JavaScript中實(shí)現(xiàn)跨標(biāo)簽頁通信的方法詳解的文章就介紹到這了,更多相關(guān)JavaScript跨標(biāo)簽頁通信內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- JavaScript中不同標(biāo)簽頁間通信的常見方式小結(jié)
- JavaScript中瀏覽器多標(biāo)簽頁通信的8種方案盤點(diǎn)
- JavaScript實(shí)現(xiàn)瀏覽器內(nèi)多個(gè)標(biāo)簽頁通信方式詳解
- JavaScript實(shí)現(xiàn)瀏覽器內(nèi)多個(gè)標(biāo)簽頁之間通信
- JavaScript使用Broadcast?Channel實(shí)現(xiàn)前端跨標(biāo)簽頁通信
- JavaScript中跨標(biāo)簽頁通信的常見方式
- JavaScript常見的跨標(biāo)簽頁通信方式總結(jié)
- JavaScript實(shí)現(xiàn)瀏覽器不同標(biāo)簽頁通信的原理與實(shí)踐
相關(guān)文章
JavaScript提高網(wǎng)站性能優(yōu)化的建議(二)
這篇文章主要介紹了JavaScript提高網(wǎng)站性能優(yōu)化的建議(二)的相關(guān)資料,需要的朋友可以參考下2016-07-07JavaScript實(shí)現(xiàn)彈出子窗口并傳值給父窗口
這篇文章主要介紹了JavaScript實(shí)現(xiàn)彈出子窗口并傳值給父窗口,方法很簡單,這里推薦給大家,需要的朋友可以參考下2014-12-12JavaScript基于自定義函數(shù)判斷變量類型的實(shí)現(xiàn)方法
這篇文章主要介紹了JavaScript基于自定義函數(shù)判斷變量類型的實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了javascript判斷變量類型的自定義函數(shù)定義與使用方法,并針對不同瀏覽器給出了相關(guān)的分析與說明,需要的朋友可以參考下2016-11-11JavaScript實(shí)現(xiàn)隨機(jī)替換圖片的方法
這篇文章主要介紹了JavaScript實(shí)現(xiàn)隨機(jī)替換圖片的方法,涉及javascript中Math.random方法的靈活運(yùn)用,需要的朋友可以參考下2015-04-04JavaScript實(shí)現(xiàn)獲取年月日時(shí)間的方法總結(jié)
這篇文章主要為大家學(xué)習(xí)介紹了JavaScript如何實(shí)現(xiàn)獲取年月日以及各種格式的時(shí)間,文中的示例代碼簡潔易懂,感興趣的小伙伴可以了解一下2023-08-08JS實(shí)現(xiàn)合并兩個(gè)數(shù)組并去除重復(fù)項(xiàng)只留一個(gè)的方法
這篇文章主要介紹了JS實(shí)現(xiàn)合并兩個(gè)數(shù)組并去除重復(fù)項(xiàng)只留一個(gè)的方法,涉及JavaScript數(shù)組合并及去重的相關(guān)技巧,需要的朋友可以參考下2015-12-12