怎样防止重复发送 Ajax 请求?
作者:长天之云
链接:https://www.zhihu.com/question/19805411/answer/15465427 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 我推荐用主动查询状态的方式(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毫秒)进行处理。 任意两次提交的间隔时间,必须大于一个指定时间,才会促成有效提交;即不给休息不干活。 module.submit = debounce(150, function() {
// todo
})
还是以知乎草稿举例,当在编辑器内按下 ctrl + s 时,可以手动保存草稿;如果你连按,程序会表示不理解为什么你要连按,只有等你放弃连按,它才会继续。 ============
而方式 C 甚至可以和方式 B 组合使用,比如自动完成组件(Google 首页的搜索就是):
----- update 2013-01-08 ----- E. 记忆型 var scrape = memoize(function(url) {
return $.post('/scraper', { 'url': url })
})
对于同样的参数,其返回始终结果是恒等的——每次都将返回同一对象。 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 ----- 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
})
G. 采样型 这是最近重构时联想到的,一种和上面都不同的去重操作,可以应用在自动加载(timeline)行为控制上: autoload.listen(feeds,'next',sample(3,function() {
this.enable()
}))
如果 sample 是固化的选择函数(n 选 1),它这实际上会这样工作: O-O-X-O-O-X
但「自动加载」的应用可能想要的是(两次自动,一次手动): X-X-O-X-X-O
对于这种情况,可以定义作为配置的选择函数来实现控制: options { sample: (n) => n % 3 !== 0 }
(编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |