问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501

如何防止异步请求的重复提交

发布网友 发布时间:2022-04-20 05:10

我来回答

1个回答

热心网友 时间:2022-04-22 18:21

不推荐用外部变量锁定或修改按钮状态的方式,因为那样比较难:

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

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

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

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

A. 独占型提交
只允许同时存在一次提交操作,并且直到本次提交完成才能进行下一次提交。mole.submit = function() {
if (this.promise_.state() === 'pending') {
return
}
return this.promise_ = $.post('/api/save')
}

B. 贪婪型提交
无*的提交,但是以最后一次操作为准;亦即需要尽快给出最后一次操作的反馈,而前面的操作结果并不重要。
mole.submit = function() {
if (this.promise_.state() === 'pending') {
this.promise_.abort()
}
// todo
}

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

C. 节制型提交
无论提交如何频繁,任意两次有效提交的间隔时间必定会大于或等于某一时间间隔;即以一定频率提交。mole.submit = throttle(150, function() {
// todo
})

如果客户发送每隔100毫秒发送过来10次请求,此模块将只接收其中6个(每个在时间线上距离为150毫秒)进行处理。
这也是解决查询冲突的一种可选手段,比如以知乎草稿举例,仔细观察可以发现:
编辑器的 blur 事件会立即触发保存;
保存按钮的 click 事件也会立即触发保存;
但是存在一种情况会使这两个事件在数毫秒内连续发生——当焦点在编辑器内部,并且直接去点击保存按钮——这时用 throttle 来处理是可行的。
另外还有一些事件处理会很频繁地使用 throttle,如: resize、scroll、mousemove。

D. 懒惰型提交
任意两次提交的间隔时间,必须大于一个指定时间,才会促成有效提交;即不给休息不干活。mole.submit = debounce(150, function() {
// todo
})

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

============
更多记忆中的例子

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

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

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

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

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

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

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

----- update 2013-01-08 -----
E. 记忆型
var scrape = memoize(function(url) {
return $.post('/scraper', { 'url': url })
})

对于同样的参数,其返回始终结果是恒等的——每次都将返回同一对象。
应用例子有编辑器,如粘贴内容时抓取其中的链接信息,memoize 用以保证同样的链接不会抓取两次。

----- update 2013-03-27 -----
F. 累积型
前几天处理自动完成事件时得到这个函数,发现也可以用在处理连续事件上,它能够把连续的多次提交合并为一个提交,比如:
var request = makePile(5, function() {
$.post('/', { list: JSON.stringify([].slice.call(arguments)) })
})

// 连续发送五次
request({a:1}), request({a:2}), request({a:3}), request({a:4}), request({a:5})
/* post =>
list:[{"a":1},{"a":2},{"a":3},{"a":4},{"a":5}]
*/

样例实现:
var makePile = function(count, onfilter, onvalue) {
var values = [], id = function(value) { return value }
return function(value) {
values.push((onvalue || id).apply(this, arguments))
if (values.length === count) {
onfilter.apply(this, values)
values = []
}
}
}

----- update 2013-04-16 -----
另一种累积是按时间而不是次数,比如应用在行为统计上,可能在瞬间收集到数十上百类似的行为,这时可以用上面 pile 的结构加上 debounce 来防止大批重复请求(但又不丢失任何统计):

var trackFactory = function(delay, action) {
var params = [], slice = [].slice
var touch = debounce(delay, function() {
if (params.length) {
action(params)
params = []
}
})
return function() {
params.push(slice.call(arguments))
touch()
}
}

var track = trackFactory(550, function(params) {
// send tracking request
})
声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com
你喜欢外柔内刚的女孩还是外刚内柔的女孩? 男人喜欢外柔内刚还是外刚内柔型的女生 ...怎样涂好颜色啊,用彩笔还是彩铅啊...谢谢啦 ...也转动了,但是里面却没有动静,也没有风不过半分钟就关了,不是... 空调开机没反应,用遥控器点开关没反应,确定不是遥控器问题,空调重新插... 请问一下 上海电信的融合宽带一个月多少钱? 本人男 脸太圆想要瘦脸 不用药 女生初涉期货行业从事什么岗位比较好 为什么较少女生选择做期货交易 我现在就读大学,现在学校里新建一个期货专业,我在考虑转专业_百度知 ... 为什么Underscore要提供JavaScript标准已有的filte... JavaScript 为什么没有分开执行,要怎么写 2020年,全栈开发者应该学些什么?(二) 如何形象的描述反应式编程中的背压机制 如何形象地描述RxJava中的背压和流控机制 2020年,全栈开发者应该学些什么?(三) javascript 防止重复点击叫什么术语 debounce 和 throttle 的区别 css3 linear-gradient线性渐变如何使用才有效果?求... css3.0有关圆角和渐变 HTML5做文字渐变的方法 CSS有没有彩色字体的样式啊?就是将一个字显示成彩... CSS3里面的线性渐变:linear-gradient参数是什么样... DW中鼠标经过字体时字体的颜色逐渐变化,CSS怎么做 css3 渐变两个颜色 各占百分之五十怎么设置 在CSS3中渐变有几种 css怎么弄文字渐变要火狐能看得见的谢谢 css3中有几种渐变 CSS3怎么做出过渡渐变效果 用css3给文字加渐变颜色,如何实现,求大神帮忙 为什么在vue 2.0 无法使用debounce 怎样防止重复发送 Ajax 请求 微信运动放在包里会计步数吗? jquery.ba-throttle-debounce.min.js怎么用 微信运动手机放在口袋里计步吗 林俊杰 醉赤壁歌词 林俊杰得《醉赤壁》歌词,谁有 林俊杰醉赤壁歌词 醉赤壁的歌词 林俊杰《醉赤壁》的歌词?谁有?? 林俊杰-醉赤壁歌词 醉赤壁的全歌词 醉赤壁 歌词 赤壁醉的歌词 赤壁赋歌词,林俊杰唱的 求林俊杰的《醉赤壁》的歌词 林俊杰醉赤壁歌词我知道,谁写一下 醉赤壁啊,歌词是什么意思 在线等 {醉赤壁} 歌词 如何评价林俊杰《醉赤壁》这首歌?