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

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

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

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


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

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

module.submit=function(){
if(this.promise_.state()==='pending'){
return
}
returnthis.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,sans-serif;font-size:13px;line-height:22.1000003814697px;">还是以知乎草稿举例,当在编辑器内按下 ctrl + s 时,可以手动保存草稿;如果你连按,程序会表示不理解为什么你要连按,只有等你放弃连按,它才会继续。
============
更多记忆中的例子
方式 C 和 方式 D 有时更加通用,比如这些情况:
  • 游戏中你捡到一把威力强大的高速武器,为了防止你的***在屏幕上打成一条直线,可以 throttle 来控制频率;

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

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


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

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

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


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

    对于同样的参数,其返回始终结果是恒等的――每次都将返回同一对象。
    应用例子有编辑器,如粘贴内容时抓取其中的链接信息,memoize 用以保证同样的链接不会抓取两次。
    ----- update 2013-03-27 -----
    F. 累积型
    前几天处理自动完成事件时得到这个函数,发现也可以用在处理连续事件上,它能够把连续的多次提交合并为一个提交,比如:
    varrequest=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}]*/

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


    ----- update 2013-04-16 -----
    另一种累积是按时间而不是次数,比如应用在行为统计上,可能在瞬间收集到数十上百类似的行为,这时可以用上面 pile 的结构加上 debounce 来防止大批重复请求(但又不丢失任何统计):
    vartrackFactory=function(delay,action){ varparams=[],slice=[].slice vartouch=debounce(delay,function(){ if(params.length){ action(params) params=[] } }) returnfunction(){ params.push(slice.call(arguments)) touch() }}vartrack=trackFactory(550,function(params){ //sendtrackingrequest})


    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}

    即每个下一次加载完成之后, 每三次有两次对下一次加载实行自动加载。

    怎么防止重复发送Ajax的更多相关文章

    1. Javascript节流函数throttle和防抖函数debounce

      这篇文章主要介绍了Javascript节流函数throttle和防抖函数debounce,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

    2. iOS编程学习中关于throttle的那些事

      这篇文章主要给大家介绍了关于iOS编程学习中throttle的那些事,文中通过示例代码介绍的非常详细,对各位iOS的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。

    3. 如何理解JS函数防抖和函数节流

      函数防抖和函数节流都是对函数进行特殊的设置,减少该函数在某一时间段内频繁触发带来的副作用。二者只是采用的设置方式和原理不一样,其最终的目的是一样的。

    4. javascript中的throttle和debounce浅析

      这篇文章主要介绍了javascript中的throttle和debounce浅析,分别介绍了throttle和debounce的使用场景及具体案例,需要的朋友可以参考下

    5. Javascript Throttle & Debounce应用介绍

      Throttle:无视一定时间内所有的调用Debounce:一定间隔内没有调用时,接下来将为大家介绍下Throttle & Debounce的应用,感兴趣的朋友可以参考下哈

    6. 怎样防止重复发送 Ajax 请求?

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

    7. 如何防止重复发送 Ajax 请求?

      module.submit=function(){if{return}returnthis.promise_=$.post}B.贪婪型提交无限制的提交,但是以最后一次操作为准;亦即需要尽快给出最后一次操作的反馈,而前面的操作结果并不重要。C.节制型提交无论提交如何频繁,任意两次有效提交的间隔时间必定会大于或等于某一时间间隔;即以一定频率提交。module.submit=throttle如果客户发送每隔100毫秒发送过来10次请求,此模块将只接收其中6个进行处理。另外还有一些事件处理会很频繁地使用throt

    8. 防止ajax多次重复请求

      B.贪婪型提交无限制的提交,但是以最后一次操作为准;亦即需要尽快给出最后一次操作的反馈,而前面的操作结果并不重要。如果客户发送每隔100毫秒发送过来10次请求,此模块将只接收其中6个进行处理。

    9. 怎么防止重复发送Ajax

      另外还有一些事件处理会很频繁地使用throttle,如:resize、scroll、mousemove。游戏中你捡到一把威力强大的高速武器,为了防止你的***在屏幕上打成一条直线,可以throttle来控制频率;在弹幕型游戏里,为了防止你把射击键夹住来进行无脑游戏,可以用debounce来控制频率;在编译任务里,守护进程监视了某一文件夹里所有的文件,但某种操作能够瞬间造成大量文件改变,这时一个简单的debounce可以使编译任务只执行一次。

    10. angular – rxjs throttle this.durationSelector不是函数

      我试图用以下代码限制ngrx存储操作更新事件这给我一个错误atThrottleSubscriber.tryDurationSelector(throttle.js:80)TypeError:this.durationSelectorisnotafunction当我用.throttle(()=>1000)替换.throttle(1000)时,它会抛出一个不同的错误,清楚地显示油门需要一个功能,而不是

    随机推荐

    1. xe-ajax-mock 前端虚拟服务

      最新版本见Github,点击查看历史版本基于XEAjax扩展的Mock虚拟服务插件;对于前后端分离的开发模式,ajax+mock使前端不再依赖后端接口开发效率更高。CDN使用script方式安装,XEAjaxMock会定义为全局变量生产环境请使用xe-ajax-mock.min.js,更小的压缩版本,可以带来更快的速度体验。

    2. vue 使用 xe-ajax

      安装完成后自动挂载在vue实例this.$ajaxCDN安装使用script方式安装,VXEAjax会定义为全局变量生产环境请使用vxe-ajax.min.js,更小的压缩版本,可以带来更快的速度体验。cdnjs获取最新版本点击浏览已发布的所有npm包源码unpkg获取最新版本点击浏览已发布的所有npm包源码AMD安装require.js安装示例ES6Module安装通过Vue.use()来全局安装示例./Home.vue

    3. AJAX POST数据中文乱码解决

      前端使用encodeURI进行编码后台java.net.URLDecoder进行解码编解码工具

    4. Koa2框架利用CORS完成跨域ajax请求

      实现跨域ajax请求的方式有很多,其中一个是利用CORS,而这个方法关键是在服务器端进行配置。本文仅对能够完成正常跨域ajax响应的,最基本的配置进行说明。这样OPTIONS请求就能够通过了。至此为止,相当于仅仅完成了预检,还没发送真正的请求呢。

    5. form提交时,ajax上传文件并更新到<input>中的value字段

    6. ajax的cache作用

      filePath="+escape;},error:{alert;}});解决方案:1.加cache:false2.url加随机数正常代码:网上高人解读:cache的作用就是第一次请求完毕之后,如果再次去请求,可以直接从缓存里面读取而不是再到服务器端读取。

    7. 浅谈ajax上传文件属性contentType = false

      默认值为contentType="application/x-www-form-urlencoded".在默认情况下,内容编码类型满足大多数情况。在这里,我们主要谈谈contentType=false.在使用ajax上传文件时:在其中先封装了一个formData对象,然后使用post方法将文件传给服务器。说到这,我们发现在JQueryajax()方法中我们使contentType=false,这不是冲突了吗?这就是因为当我们在form标签中设置了enctype=“multipart/form-data”,

    8. 909422229_ajaxFileUpload上传文件

      ajaxFileUpload.js很多同名的,因为做出来一个很容易。我上github搜AjaxFileUpload出来很多类似js。ajaxFileUpload是一个异步上传文件的jQuery插件传一个不知道什么版本的上来,以后不用到处找了。语法:$.ajaxFileUploadoptions参数说明:1、url上传处理程序地址。2,fileElementId需要上传的文件域的ID,即的ID。3,secureuri是否启用安全提交,默认为false。4,dataType服务器返回的数据类型。6,error

    9. AJAX-Cache:一款好用的Ajax缓存插件

      原文链接AJAX-Cache是什么Ajax是前端开发必不可少的数据获取手段,在频繁的异步请求业务中,我们往往需要利用“缓存”提升界面响应速度,减少网络资源占用。AJAX-Cache是一款jQuery缓存插件,可以为$.ajax()方法扩展缓存功能。

    10. jsf – Ajax update/render在已渲染属性的组件上不起作用

      我试图ajax更新一个有条件渲染的组件。我可以确保#{user}实际上是可用的。这是怎么引起的,我该如何解决呢?必须始终在ajax可以重新呈现之前呈现组件。Ajax正在使用JavaScriptdocument.getElementById()来查找需要更新的组件。但是如果JSF没有将组件放在第一位,那么JavaScript找不到要更新的内容。解决方案是简单地引用总是渲染的父组件。

    返回
    顶部