麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 編程 > JSP > 正文

在Web關(guān)閉頁面時發(fā)送Ajax請求的實現(xiàn)方法

2024-09-05 00:23:35
字體:
供稿:網(wǎng)友

前言

有時候我們需要在用戶離開頁面的時候,做一些上報來記錄用戶行為。又或者是發(fā)送服務(wù)器ajax請求,通知服務(wù)器用戶已經(jīng)離開,比如直播間內(nèi)的退房操作。

本文主要分兩部分來講解怎么完成退出行為的上報。

1.事件監(jiān)聽

瀏覽器有兩個事件可以用來監(jiān)聽頁面關(guān)閉,beforeunload和unload。

beforeunload是在文檔和資源將要關(guān)閉的時候調(diào)用的, 這時候文檔還是可見的,并且在這個關(guān)閉的事件還是可以取消的。比如下面這種寫法就會讓用戶導(dǎo)致在刷新或者關(guān)閉頁面時候,有個彈窗提醒用戶是否關(guān)閉。

window.addEventListener("beforeunload", function (event) { // Cancel the event as stated by the standard. event.preventDefault(); // Chrome requires returnValue to be set. event.returnValue = '';});

unload則是在頁面已經(jīng)正在被卸載時發(fā)生,此時文檔所處的狀態(tài)是:1.所有資源仍存在(圖片,iframe等);2.對于用戶所有資源不可見;3.界面交互無效(window.open, alert, confirm 等);4.錯誤不會停止卸載文檔的過程。

基于以上兩個方法就可以實現(xiàn)對頁面關(guān)閉的事件監(jiān)聽了,為了穩(wěn)妥,可以兩個事件都監(jiān)聽。然后對監(jiān)聽函數(shù)做處理,讓關(guān)閉事件只調(diào)用一次。

2.請求發(fā)送

有了上面的監(jiān)聽,事情只完成了一半,如果我們在監(jiān)聽中直接發(fā)送ajax請求,就會發(fā)現(xiàn)請求被瀏覽器abort了,無法發(fā)送出去。在頁面卸載的時候,瀏覽器并不能保證異步的請求能夠成功發(fā)出去。

我們有幾種方式可以解決這個問題:

方案1: 發(fā)送同步的ajax請求

var oAjax = new XMLHttpRequest();oAjax.open('POST', url + '/user/register', false);//false表示同步請求oAjax.setRequestHeader("Content-type", "application/x-www-form-urlencoded");oAjax.onreadystatechange = function() { if (oAjax.readyState == 4 && oAjax.status == 200) { var data = JSON.parse(oAjax.responseText); } else { console.log(oAjax); }};oAjax.send('a=1&b=2');

這種方式雖然有效,但是用戶需要等待請求結(jié)束才可以關(guān)閉頁面。對用戶的體驗不好。

方案2:發(fā)送異步請求,并且在服務(wù)端忽略ajax的abort

雖然異步請求會被瀏覽器abort,但是如果服務(wù)端可以忽略abort,仍然正常執(zhí)行,也是可以的。比如PHP有ignore_user_abort函數(shù)可以忽略abort。這樣需要改造后臺,一般不太可行..

方案3:使用navigator.sendBeacon發(fā)送異步請求

根據(jù)MDN的介紹:

這個方法主要用于滿足 統(tǒng)計和診斷代碼 的需要,這些代碼通常嘗試在卸載(unload)文檔之前向web服務(wù)器發(fā)送數(shù)據(jù)。過早的發(fā)送數(shù)據(jù)可能導(dǎo)致錯過收集數(shù)據(jù)的機會。然而, 對于開發(fā)者來說保證在文檔卸載期間發(fā)送數(shù)據(jù)一直是一個困難。因為用戶代理通常會忽略在卸載事件處理器中產(chǎn)生的異步 XMLHttpRequest 。

從介紹上可以看出,這個方法就是用來在用戶離開時發(fā)請求的。非常適合這種場景。

使用方式是這樣的:

navigator.sendBeacon(url [, data]);

sendBeacon支持發(fā)送的data可以是ArrayBufferView, Blob, DOMString, 或者 FormData 類型的數(shù)據(jù)。

下面是幾種使用sendBeacon發(fā)送請求的方式,可以修改header和內(nèi)容的格式,因為一般和服務(wù)器的通信方式都是固定的,如果修改了header或者內(nèi)容,服務(wù)器就無法正常識別出來了。

(1)使用Blob來發(fā)送

使用blob發(fā)送的好處是可以自己定義內(nèi)容的格式和header。比如下面這種設(shè)置方式,就是可以設(shè)置content-type為application/x-

blob = new Blob([`room_id=123`], {type : 'application/x-www-form-urlencoded'});navigator.sendBeacon("/cgi-bin/leave_room", blob);

Web,發(fā)送,Ajax

(2)使用FormData對象,但是這時content-type會被設(shè)置成"multipart/form-data"。

var fd = new FormData();fd.append('room_id', 123);navigator.sendBeacon("/cgi-bin/leave_room", fd);

Web,發(fā)送,Ajax

(3)數(shù)據(jù)也可以使用URLSearchParams 對象,content-type會被設(shè)置成"text/plain;charset=UTF-8" 。

var params = new URLSearchParams({ room_id: 123 })navigator.sendBeacon("/cgi-bin/leave_room", params);

Web,發(fā)送,Ajax

通過嘗試,可以發(fā)現(xiàn)使用blob發(fā)送比較方便,內(nèi)容的設(shè)置也比較靈活,如果發(fā)送的消息抓包后發(fā)現(xiàn)后臺沒有識別出來,可以嘗試修改內(nèi)容的string或者header,來找到合適的方式發(fā)送請求。
參考鏈接:

  • sendBeacon API not working temporarily due to security issue, any workaround?
  • Sending AJAX Data when User Moves Away / Exits from Page
  • Setting HTTP Headers in a Beacon Request

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網(wǎng)的支持。


注:相關(guān)教程知識閱讀請移步到JSP教程頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 激情夜色 | 黄色大片免费看 | 美女在线观看视频一区二区 | 色中色激情影院 | 九九精品影院 | 蜜桃网站在线观看 | 成人午夜亚洲 | 91成人免费视频 | 久久影院yy6080 | 亚洲一区在线免费视频 | 精品国产一区二区三区久久久蜜月 | 爽爽视频免费看 | 午色影院 | 特级西西444www大精品视频免费看 | 日本黄色a视频 | 第一区免费在线观看 | 在线免费观看欧美 | 国色天香综合网 | 国产91精品久久久久久久 | 欧美黄 片免费观看 | 亚洲精品无码不卡在线播放he | 成人免费乱码大片a毛片视频网站 | 国产精品美女一区二区 | 黄色特级一级片 | 中国免费黄色 | 国产一区二区不卡 | 欧美顶级毛片在线播放小说 | 久久99亚洲精品久久99果 | 国产精品久久久久久久模特 | 国产高潮好爽受不了了夜色 | 黄色av网站免费 | 国产精品99爱| 黄色毛片a级 | 国产精品自拍啪啪 | 久久精品一区二区三区四区五区 | 视频一区二区中文字幕 | 精品国产一区二区三区成人影院 | 欧美女优一区 | 国产精品久久久久久久hd | av观看国产| 久久精品亚洲一区二区三区观看模式 |