成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何在頁面關閉或跳轉時優雅的發送Ajax請求

開發 前端
有時候我們需要在用戶離開頁面的時候,做一些上報來記錄用戶行為或者埋點,又或者是發送到服務器的ajax請求。那如何保證請求能夠正確的送達就是一個很關鍵的點。

[[282720]]

有時候我們需要在用戶離開頁面的時候,做一些上報來記錄用戶行為或者埋點,又或者是發送到服務器的ajax請求。那如何保證請求能夠正確的送達就是一個很關鍵的點。下面我們就來介紹下應該如何操作:

首先,要做事件監聽

瀏覽器有兩個事件可以用來監聽頁面關閉,beforeunload和unload。

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

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

unload則是在頁面已經正在被卸載時發生,此時文檔所處的狀態是:

1.所有資源仍存在(圖片,iframe等);

2.對于用戶所有資源不可見;

3.界面交互無效(window.open, alert, confirm 等);

4.錯誤不會停止卸載文檔的過程。

基于以上兩個方法就可以實現對頁面關閉的事件監聽了,為了穩妥,可以兩個事件都監聽。然后對監聽函數做處理,讓關閉事件只調用一次,比如用變量控制請求發送的次數。

其次,發送請求

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

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

方案1: 發送同步的ajax請求

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

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

方案2:發送異步請求,并且在服務端忽略ajax的abort

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

方案3:使用navigator.sendBeacon發送異步請求

根據MDN的介紹:

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

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

  1. navigator.sendBeacon(url [, data]); 

sendBeacon支持發送的data可以是ArrayBufferView, Blob, DOMString, 或者 FormData 類型的數據。

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

(1)使用Blob來發送 使用blob發送的好處是可以自己定義內容的格式和header。比如下面這種設置方式,就是可以設置content-type為application/x-www-form-urlencoded。

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

如何在頁面關閉或跳轉時優雅的發送Ajax請求

 

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

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

如何在頁面關閉或跳轉時優雅的發送Ajax請求

 

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

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

如何在頁面關閉或跳轉時優雅的發送Ajax請求

 

通過嘗試,可以發現使用blob發送比較方便,內容的設置也比較靈活,如果發送的消息抓包后發現后臺沒有識別出來,可以嘗試修改內容的string或者header,來找到合適的方式發送請求。 

責任編輯:龐桂玉 來源: 今日頭條
相關推薦

2022-07-03 17:55:53

HTTP頁面瀏覽器

2023-07-30 22:25:00

JavaScrip服務端Web

2024-03-29 00:00:00

JSAPI網絡

2021-01-19 10:35:49

JVM場景函數

2019-01-27 14:37:47

數據HTTP服務

2020-03-27 15:10:23

SpringJava框架

2022-04-11 08:17:07

JVMJava進程

2021-03-28 09:17:18

JVM場景鉤子函數

2020-06-04 17:00:37

Linux命令腳本

2021-06-17 09:32:39

重復請求并發請求Java

2017-12-19 10:03:44

JavaLinux代碼

2024-11-13 16:37:00

Java線程池

2022-03-24 14:49:57

HTTP前端

2009-07-01 18:08:18

JSP頁面跳轉

2018-02-24 16:30:49

LinuxUnixBash

2021-12-06 09:57:25

容器Linux信號

2022-03-07 07:33:24

Spring自定義機制線程池

2021-09-27 13:16:24

LinuxWindows游戲

2017-03-10 10:37:16

Linux命令腳本

2021-01-21 08:17:23

Linux電池充放電狀態
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文字幕在线看第二 | 成人三级视频 | 国产精品乱码一区二区三区 | 亚洲 中文 欧美 日韩 在线观看 | 中文字幕日韩欧美 | 国内精品视频在线观看 | 国产精品无 | 福利av在线 | 国产精品爱久久久久久久 | 日日操天天射 | 青青草一区二区三区 | 中文字幕亚洲欧美 | 人人爽人人爽人人片av | 欧美福利| 午夜天堂精品久久久久 | 伊人狼人影院 | 欧美黄色片 | 一区二区在线 | 精品一级 | 91精品国产91久久久久游泳池 | 欧美日韩专区 | 日本在线小视频 | 99热成人在线 | 国产激情福利 | 国产在线视频网 | 精品福利av导航 | 最新中文字幕第一页视频 | 秋霞av国产精品一区 | 亚洲精品一区二区三区在线观看 | 中文字幕乱码一区二区三区 | 日本高清在线一区 | 国产亚洲成av人片在线观看桃 | 亚洲国产精品一区二区第一页 | 国产视频第一页 | 91社区视频 | 成人免费视频网站在线看 | 妞干网av | 欧美一区二区三区电影 | 中文字幕视频在线看 | 日韩欧美网 | 日韩中文字幕视频 |