防止重复发送 Ajax 请求


Posted in Javascript onFebruary 15, 2017

要考虑并理解 success, complete, error, timeout 这些事件的区别,并注册正确的事件,一旦失误,功能将不再可用;

不可避免地比普通流程要要多注册一个 complete 事件;

恢复状态的代码很容易和不相干的代码混合在一起;

推荐用主动查询状态的方式(A、B,jQuery 为例)或工具函数的方式(C、D)来去除重复操作,并提供一些例子作为参考:

A. 独占型提交

只允许同时存在一次提交操作,并且直到本次提交完成才能进行下一次提交。

module.submit = function() {
 if (this.promise_.state() === 'pending') {
 return
 }
 return this.promise_ = $.post('/api/save')
}

B. 贪婪型提交

无限制的提交,但是以最后一次操作为准;亦即需要尽快给出最后一次操作的反馈,而前面的操作结果并不重要。

module.submit = function() {
 if (this.promise_.state() === 'pending') {
 this.promise_.abort()
 }
 // todo
}

比如某些应用的条目中,有一些进行类似「喜欢」或「不喜欢」操作的二态按钮。如果按下后不立即给出反馈,用户的目光焦点就可能在那个按钮上停顿许久;如果按下时即时切换按钮的状态,再在程序上用 abort 来实现积极的提交,这样既能提高用户体验,还能降低服务器压力,皆大欢喜。

C. 节制型提交

无论提交如何频繁,任意两次有效提交的间隔时间必定会大于或等于某一时间间隔;即以一定频率提交。

module.submit = throttle(150, function() {
 // todo
})

如果客户发送每隔100毫秒发送过来10次请求,此模块将只接收其中6个(每个在时间线上距离为150毫秒)进行处理。

这也是解决查询冲突的一种可选手段,比如以知乎草稿举例,仔细观察可以发现:

编辑器的 blur 事件会立即触发保存;

保存按钮的 click 事件也会立即触发保存;

但是存在一种情况会使这两个事件在数毫秒内连续发生——当焦点在编辑器内部,并且直接去点击保存按钮——这时用 throttle 来处理是可行的。

另外还有一些事件处理会很频繁地使用 throttle,如: resize、scroll、mousemove。

D. 懒惰型提交

任意两次提交的间隔时间,必须大于一个指定时间,才会促成有效提交;即不给休息不干活。

module.submit = debounce(150, function() {
 // todo
})

还是以知乎草稿举例,当在编辑器内按下 ctrl + s 时,可以手动保存草稿;如果你连按,程序会表示不理解为什么你要连按,只有等你放弃连按,它才会继续。

============

更多记忆中的例子

方式 C 和 方式 D 有时更加通用,比如这些情况:

游戏中你捡到一把威力强大的高速武器,为了防止你的子弹在屏幕上打成一条直线,可以 throttle 来控制频率;

在弹幕型游戏里,为了防止你把射击键夹住来进行无脑游戏,可以用 debounce 来控制频率;

在编译任务里,守护进程监视了某一文件夹里所有的文件(如任一文件的改变都可以触发重新编译,一次执行就需要2秒),但某种操作能够瞬间造成大量文件改变(如 git checkout),这时一个简单的 debounce 可以使编译任务只执行一次。

而方式 C 甚至可以和方式 B 组合使用,比如自动完成组件(Google 首页的搜索就是):

当用户快速输入文本时(特别是打字能手),可以 throttle keypress 事件处理函数,以指定时间间隔来提取文本域的值,然后立即进行新的查询;

当新的查询需要发送,但上一个查询还没返回结果时,可以 abort 未完成的查询,并立即发送新查询;

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持三水点靠木!

Javascript 相关文章推荐
表单(FORM)的一些实用效果代码
Mar 25 Javascript
可以用来调试JavaScript错误的解决方案
Aug 07 Javascript
JavaScript中的几个关键概念的理解-原型链的构建
May 12 Javascript
图片延迟加载的实现代码(模仿懒惰)
Mar 29 Javascript
window.print打印指定div指定网页指定区域的方法
Aug 04 Javascript
jQuery插件slick实现响应式移动端幻灯片图片切换特效
Apr 12 Javascript
深入浅出分析javaScript中this用法
May 09 Javascript
js实现旋转木马效果
Mar 17 Javascript
bootstrap+jQuery 实现下拉菜单中复选框全选和全不选效果
Jun 12 jQuery
vue做网页开场视频的实例代码
Oct 20 Javascript
Node.js API详解之 net模块实例分析
May 18 Javascript
微信小程序实现列表左右滑动
Nov 19 Javascript
Input文本框随着输入内容多少自动延伸的实现
Feb 15 #Javascript
利用Jquery实现几款漂亮实用的时间轴(附示例代码)
Feb 15 #Javascript
js记录点击某个按钮的次数-刷新次数为初始状态的实例
Feb 15 #Javascript
Canvas 绘制粒子动画背景
Feb 15 #Javascript
jQuery插件jquery.kxbdmarquee.js实现无缝滚动效果
Feb 15 #Javascript
js控制按钮,防止频繁点击响应的实例
Feb 15 #Javascript
js代码延迟一定时间后执行一个函数的实例
Feb 15 #Javascript
You might like
最令PHP初学者头痛的十四个问题
2006/07/12 PHP
PHP中register_globals参数为OFF和ON的区别(register_globals 使用详解)
2012/02/05 PHP
解析php curl_setopt 函数的相关应用及介绍
2013/06/17 PHP
php实现mysql数据库操作类分享
2014/02/14 PHP
PHP代码优化之成员变量获取速度对比
2014/02/28 PHP
学习php设计模式 php实现策略模式(strategy)
2015/12/07 PHP
PHP高精确度运算BC函数库实例详解
2017/08/15 PHP
基于jquery的无刷新分页技术
2011/06/11 Javascript
javascript实现图片左右滚动效果【可自动滚动,有左右按钮】
2016/09/19 Javascript
Ajax异步获取html数据中包含js方法无效的解决方法
2017/02/20 Javascript
jQuery中Chosen三级联动功能实例代码
2017/03/07 Javascript
layer弹出层框架alert与msg详解
2017/03/14 Javascript
vue实现组件之间传值功能示例
2018/07/13 Javascript
jQuery移动端跑马灯抽奖特效升级版(抽奖概率固定)实现方法
2019/01/18 jQuery
原生JS实现的跳一跳小游戏完整实例
2019/01/27 Javascript
Vue Echarts实现可视化世界地图代码实例
2019/05/07 Javascript
微信小程序实现写入读取缓存详解
2019/08/30 Javascript
js事件机制----捕获与冒泡机制实例分析
2020/05/22 Javascript
JavaScript this指向相关原理及实例解析
2020/07/10 Javascript
[50:04]DOTA2上海特级锦标赛D组小组赛#2 Liquid VS VP第二局
2016/02/28 DOTA
[01:07:20]DOTA2-DPC中国联赛 正赛 Dynasty vs XG BO3 第二场 2月2日
2021/03/11 DOTA
简单介绍Python中的len()函数的使用
2015/04/07 Python
初步认识Python中的列表与位运算符
2015/10/12 Python
浅谈使用Python内置函数getattr实现分发模式
2018/01/22 Python
Python容器使用的5个技巧和2个误区总结
2019/09/26 Python
Django中使用haystack+whoosh实现搜索功能
2019/10/08 Python
python同时遍历两个list用法说明
2020/05/02 Python
python批量修改文件名的示例
2020/09/27 Python
4款Python 类型检查工具,你选择哪个呢?
2020/10/30 Python
html5+css如何实现中间大两头小的轮播效果
2018/12/06 HTML / CSS
美国购车网站:TrueCar
2016/10/19 全球购物
财产公证书样本
2014/04/04 职场文书
项目验收申请报告
2015/05/15 职场文书
学校学期工作总结
2015/08/13 职场文书
PHP实现考试倒计时功能代码
2021/04/16 PHP
SpringBoot详解自定义Stater的应用
2022/07/15 Java/Android