如何在页面关闭或跳转时优雅的发送Ajax请求
时间:2025-11-05 07:20:17 出处:应用开发阅读(143)

有时候我们需要在用户离开页面的何页时候,做一些上报来记录用户行为或者埋点,面关又或者是闭或发送到服务器的ajax请求。那如何保证请求能够正确的跳转送达就是一个很关键的点。下面我们就来介绍下应该如何操作:
首先,时优要做事件监听
浏览器有两个事件可以用来监听页面关闭,发送beforeunload和unload。请求
beforeunload是何页在文档和资源将要关闭的时候调用的, 这时候文档还是面关可见的,并且在这个关闭的闭或事件还是可以取消的。比如下面这种写法就会让用户导致在刷新或者关闭页面时候,跳转有个弹窗提醒用户是时优否关闭。
window.addEventListener("beforeunload",发送 function (event) { // Cancel the event as stated by the standard. event.preventDefault(); // Chrome requires returnValue to be set. event.returnValue = ; });unload则是在页面已经正在被卸载时发生,此时文档所处的请求状态是:
1.所有资源仍存在(图片,iframe等);
2.对于用户所有资源不可见;
3.界面交互无效(window.open,何页 alert, confirm 等);
4.错误不会停止卸载文档的过程。b2b供应网
基于以上两个方法就可以实现对页面关闭的事件监听了,为了稳妥,可以两个事件都监听。然后对监听函数做处理,让关闭事件只调用一次,比如用变量控制请求发送的次数。
其次,发送请求
有了上面的监听,事情只完成了一半,如果我们在监听中直接发送ajax请求,就会发现请求被浏览器abort了,无法发送出去。在页面卸载的时候,浏览器并不能保证异步的请求能够成功发出去。
我们有几种方式可以解决这个问题:
方案1: 发送同步的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);这种方式虽然有效,但是用户需要等待请求结束才可以关闭页面。对用户的体验不好。
方案2:发送异步请求,并且在服务端忽略ajax的abort
虽然异步请求会被浏览器abort,但是如果服务端可以忽略abort,仍然正常执行,也是可以的高防服务器。比如PHP有ignore_user_abort函数可以忽略abort。这样需要改造后台,一般不太可行。
方案3:使用navigator.sendBeacon发送异步请求
根据MDN的介绍:
这个方法主要用于满足 统计和诊断代码 的需要,这些代码通常尝试在卸载(unload)文档之前向web服务器发送数据。过早的发送数据可能导致错过收集数据的机会。然而, 对于开发者来说保证在文档卸载期间发送数据一直是一个困难。因为用户代理通常会忽略在卸载事件处理器中产生的异步 XMLHttpRequest 。
从介绍上可以看出,这个方法就是用来在用户离开时发请求的。非常适合这种场景。 使用方式是这样的:
navigator.sendBeacon(url [, data]);sendBeacon支持发送的data可以是ArrayBufferView, Blob, DOMString, 或者 FormData 类型的数据。
下面是几种使用sendBeacon发送请求的方式,可以修改header和内容的格式,因为一般和服务器的亿华云计算通信方式都是固定的,如果修改了header或者内容,服务器就无法正常识别出来了。
(1)使用Blob来发送 使用blob发送的好处是可以自己定义内容的格式和header。比如下面这种设置方式,就是可以设置content-type为application/x-www-form-urlencoded。
blob = new Blob([`room_id=123`], {type : application/x-www-form-urlencoded}); navigator.sendBeacon("/cgi-bin/leave_room", blob);
(2)使用FormData对象,但是这时content-type会被设置成"multipart/form-data"。
var fd = new FormData(); fd.append(room_id, 123); navigator.sendBeacon("/cgi-bin/leave_room", fd);
(3)数据也可以使用URLSearchParams 对象,content-type会被设置成"text/plain;charset=UTF-8" 。
var params = new URLSearchParams({ room_id: 123 }) navigator.sendBeacon("/cgi-bin/leave_room", params);
通过尝试,可以发现使用blob发送比较方便,内容的设置也比较灵活,如果发送的消息抓包后发现后台没有识别出来,可以尝试修改内容的string或者header,来找到合适的方式发送请求。
猜你喜欢
- 操作系统安装教程(一步步教你安装操作系统,让电脑焕发新生)
- 探索以8512镜头的拍摄效果与技巧(挖掘8512镜头的魅力,拍摄出精彩的作品)
- GoogleFiber(探索GoogleFiber的高速网络和创新技术)
- 探索JBL万花筒(揭秘JBL万花筒的技术与魅力)
- OPPOR9Plus在VR方面的表现如何?(探索OPPOR9Plus在虚拟现实技术应用上的优势与不足)
- 保时捷送的手表(以保时捷为代表的高档手表,融合了汽车工艺与制表技术的巅峰之作)
- 威索尼克GR07耳机(深度解析GR07耳机的卓越音质和舒适度)
- AOCG2770PF(细腻画质、流畅画面、不可忽视的价格)
- iCloud照片删除指南(掌握iCloud照片删除的方法,释放存储空间)