连续同源异步操作队列

广告位招租
扫码页面底部二维码联系

问题的发现【关注微信公众号:wwwtangshuangnet】

本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。

来自同源的多个异步操作可能引起异步冲突问【本文受版权保护】未经授权,禁止复制转载。题,特别是在网络请求时。同源操作产生了两未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。个ajax请求,它们的请求结果将用于渲染【转载请注明来源】【版权所有】唐霜 www.tangshuang.net同一个区域,然而由于网络问题,先发出的请原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。求后返回,导致最终得到的界面是错误的。

【原创不易,请尊重版权】未经授权,禁止复制转载。原创内容,盗版必究。

解决这个问题的最好办法,是利用原生XHR原创内容,盗版必究。【作者:唐霜】的abort方法,在后一次操作时,将前一转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.net次操作引起的ajax请求给cancel掉转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。

【未经授权禁止转载】【转载请注明来源】转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。

但是在现实条件下,异步操作并非都有can【版权所有,侵权必究】【访问 www.tangshuang.net 获取更多精彩内容】cel操作。js原生的Promise没有原创内容,盗版必究。原创内容,盗版必究。,原生的fetch基于Promise也没著作权归作者所有,禁止商业用途转载。【作者:唐霜】有。基于Promise的很多工具都没有c本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。ancel操作。这种情况下怎么解决这个问【访问 www.tangshuang.net 获取更多精彩内容】【原创内容,转载请注明出处】题呢?

著作权归作者所有,禁止商业用途转载。本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。

其实方法是有的,就是直接丢弃Promis著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。e的推送,不执行它的resolve回调即【原创不易,请尊重版权】【原创不易,请尊重版权】可。这样,虽然异步操作已经执行了,但不会【未经授权禁止转载】【版权所有,侵权必究】对现有的环境造成任何副作用。(虽然这样看本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】上去浪费了异步操作这个资源。)

原创内容,盗版必究。【转载请注明来源】本文版权归作者所有,未经授权不得转载。

问题的思考未经授权,禁止复制转载。

【版权所有】唐霜 www.tangshuang.net【本文受版权保护】本文版权归作者所有,未经授权不得转载。

如何来判断是否要丢掉它的回调?我们可以创【版权所有,侵权必究】【本文受版权保护】建一个队列,每次产生一个异步操作时,就将【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net它加入到队列中,当队列中存在操作对象时,【原创不易,请尊重版权】本文作者:唐霜,转载请注明出处。每次只取最后一个,等待它推送结果,执行它著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net的回调,排在它前面的操作全部丢弃掉。

【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net

基于这样的想法,我写了一个工具defer【版权所有】唐霜 www.tangshuang.net【关注微信公众号:wwwtangshuangnet】er-queue。它为异步操作创建队列,本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。并根据不同的场景实现不同的队列操作形式。【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net你可以通过这里阅读它的源码和文档【版权所有】唐霜 www.tangshuang.net。它提供了4种可供选择的场景:著作权归作者所有,禁止商业用途转载。

【转载请注明来源】著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net【未经授权禁止转载】

并联著作权归作者所有,禁止商业用途转载。

【关注微信公众号:wwwtangshuangnet】【未经授权禁止转载】【转载请注明来源】转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。

异步操作进入队列的时候立即开始执行,它们【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。的回调将按照它们进入队列的顺序执行。

【本文首发于唐霜的博客】本文版权归作者所有,未经授权不得转载。【作者:唐霜】【访问 www.tangshuang.net 获取更多精彩内容】

本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。【原创不易,请尊重版权】

这个图中有这些元素:【未经授权禁止转载】

【本文首发于唐霜的博客】转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】
  • 时间轴未经授权,禁止复制转载。
  • 著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】【原创不易,请尊重版权】
  • 圆圈:一个异步操作被push进队列【访问 www.tangshuang.net 获取更多精彩内容】
  • 【本文首发于唐霜的博客】【转载请注明来源】原创内容,盗版必究。【本文受版权保护】
  • 向下箭头:队列中开始执行某个异步操作本文版权归作者所有,未经授权不得转载。
  • 转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。
  • 虚线:执行顺序(执行流)【原创内容,转载请注明出处】
  • 【版权所有,侵权必究】【作者:唐霜】著作权归作者所有,禁止商业用途转载。【原创不易,请尊重版权】本文作者:唐霜,转载请注明出处。
  • 方框:每个异步操作的回调【原创不易,请尊重版权】
  • 【原创不易,请尊重版权】原创内容,盗版必究。【版权所有,侵权必究】【原创不易,请尊重版权】【作者:唐霜】
  • 竖线:队列周期(从开始到停止的过程)分割本文作者:唐霜,转载请注明出处。【关注微信公众号:wwwtangshuangnet】线
  • 著作权归作者所有,禁止商业用途转载。【未经授权禁止转载】【转载请注明来源】【未经授权禁止转载】【转载请注明来源】

上图表现的是,1-4这4个异步操作在不同【原创不易,请尊重版权】【转载请注明来源】的事件点被加入到队列中。它们在一个周期内【原创不易,请尊重版权】【原创不易,请尊重版权】,也就是从1开始执行,到队列中所有的异步【版权所有,侵权必究】本文作者:唐霜,转载请注明出处。操作执行完毕,进入停滞状态。1-4这些操【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。作,一进入队列就开始执行。它们执行完之后本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。,就会告知队列执行完毕状态,回调就会执行本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。。但是由于回调一定是按照异步操作进入队列转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。的顺序执行的,因此,即使4已经完成了,但原创内容,盗版必究。转载请注明出处:www.tangshuang.net它的回调也必须等到3结束后才会执行。

【转载请注明来源】转载请注明出处:www.tangshuang.net转载请注明出处:www.tangshuang.net【作者:唐霜】

在deferer-queue中,所有的回【访问 www.tangshuang.net 获取更多精彩内容】本文作者:唐霜,转载请注明出处。调一定遵循先进队列先执行,后进队列后执行未经授权,禁止复制转载。【本文首发于唐霜的博客】的顺序规则。

转载请注明出处:www.tangshuang.net【未经授权禁止转载】转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。

串联【作者:唐霜】

【访问 www.tangshuang.net 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net

异步操作进入队列后并不立即执行,它从队列【作者:唐霜】【原创不易,请尊重版权】中取出第一个,执行,等待这个异步操作完成著作权归作者所有,禁止商业用途转载。本文版权归作者所有,未经授权不得转载。(成功或失败都算完成)并执行它的回调,然本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。后再取出下一个,继续执行。

本文作者:唐霜,转载请注明出处。【转载请注明来源】本文版权归作者所有,未经授权不得转载。

【访问 www.tangshuang.net 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net

上图表现的是,1-4这4个异步操作,进入著作权归作者所有,禁止商业用途转载。【版权所有,侵权必究】队列后,按照顺序依次执行。但是,在一个周本文作者:唐霜,转载请注明出处。【作者:唐霜】期内,只有1是立即执行的,其他操作必须等本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。到上一个操作结束(promise res【未经授权禁止转载】未经授权,禁止复制转载。olved或rejected)之后,才会【本文首发于唐霜的博客】【转载请注明来源】开始执行。

【原创内容,转载请注明出处】【转载请注明来源】【版权所有,侵权必究】【转载请注明来源】

尾调【转载请注明来源】

本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。【原创内容,转载请注明出处】

异步操作进入队列后并不立即执行,它从队列本文版权归作者所有,未经授权不得转载。【关注微信公众号:wwwtangshuangnet】中取出最后一个,执行,并把前面所有的操作著作权归作者所有,禁止商业用途转载。著作权归作者所有,禁止商业用途转载。丢弃,这个操作执行完后直接进入回调,队列【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】结束。

【本文受版权保护】著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net【本文受版权保护】

【未经授权禁止转载】转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。【原创不易,请尊重版权】

这张图表现的是,1-4这个4个异步操作被【版权所有】唐霜 www.tangshuang.net【原创不易,请尊重版权】按顺序加入到队列中,但是,在一个周期内,未经授权,禁止复制转载。【关注微信公众号:wwwtangshuangnet】只有4(最后一个)被执行。1-3全部被丢【本文首发于唐霜的博客】著作权归作者所有,禁止商业用途转载。掉了。你可能会问,假如4进入队列的时候,著作权归作者所有,禁止商业用途转载。原创内容,盗版必究。1已经在执行了怎么办?答案是,1会被丢掉【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。,只要在一个周期内,只会有一个执行的回调著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。会被执行。也就是说,假如4开始执行之后,【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】突然5插入进来了,那么4会被直接丢掉,这【关注微信公众号:wwwtangshuangnet】【本文受版权保护】一个周期以最后一个回调函数被执行而结束。

【本文受版权保护】【版权所有】唐霜 www.tangshuang.net【本文受版权保护】【未经授权禁止转载】

连续【版权所有】唐霜 www.tangshuang.net

【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】

异步操作进入队列后并不立即执行,它从队列未经授权,禁止复制转载。【原创不易,请尊重版权】中取出第一个,执行,等待这个异步操作完成【本文受版权保护】转载请注明出处:www.tangshuang.net(成功或失败都算完成)并执行它的回调,之【原创不易,请尊重版权】未经授权,禁止复制转载。后取出最新的一个执行,执行完之后再看队列【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】中是否有新的,如果有新的,再取出最新的一【版权所有,侵权必究】本文作者:唐霜,转载请注明出处。个,继续执行,如此往复下去。

【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。【本文受版权保护】本文作者:唐霜,转载请注明出处。

原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。【本文首发于唐霜的博客】

这幅图表面的是一种连续不断的执行队列的模【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。式,当1-4被push进队列之后,1被执转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】行,2-4被丢掉。当1完成之后,检查队列【访问 www.tangshuang.net 获取更多精彩内容】【本文首发于唐霜的博客】是否加入了新的异步操作,如果检查到5-6本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】,那么取6进行执行,把5丢掉。进入后续过未经授权,禁止复制转载。【未经授权禁止转载】程后,感觉上和尾调非常像,但是,尾调是在【原创不易,请尊重版权】转载请注明出处:www.tangshuang.net一个周期内,只会执行最后一个回调,而连续【版权所有】唐霜 www.tangshuang.net【未经授权禁止转载】不是,它会不断的检查队列是否已经被执行完原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。,如果没有,那么就会执行一个类似尾调的过【访问 www.tangshuang.net 获取更多精彩内容】【原创不易,请尊重版权】程,这个过程结束,又会再做一次。队列开始【本文受版权保护】【转载请注明来源】的时候,取第一个异步操作执行,当第一个操转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.net作结束后,后面之后去取最后一个执行。这些【原创不易,请尊重版权】【未经授权禁止转载】执行之间是串联关系,只有正在执行的异步操【作者:唐霜】【原创不易,请尊重版权】作已经完成的情况下,新的操作才会被执行。【本文受版权保护】【版权所有,侵权必究】这是一种比较奇葩的场景,但是它可以解决一著作权归作者所有,禁止商业用途转载。原创内容,盗版必究。些需要连续不断做某些事情的应用。

著作权归作者所有,禁止商业用途转载。【原创不易,请尊重版权】【未经授权禁止转载】

注意:当一个周期结束后,队列处于停滞状态【本文受版权保护】【本文首发于唐霜的博客】,但是,一旦有新的defer被push进【本文受版权保护】【作者:唐霜】队列,它又会开始一个新的周期。

本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net

问题的解决【版权所有】唐霜 www.tangshuang.net

本文版权归作者所有,未经授权不得转载。【作者:唐霜】【原创不易,请尊重版权】【访问 www.tangshuang.net 获取更多精彩内容】

基于上述的思考,我最终发布了defere【原创不易,请尊重版权】原创内容,盗版必究。r-queue,你可以通过npm安装这个【本文受版权保护】【原创不易,请尊重版权】包,在自己的项目中使用它。它的操作模式超本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】级简单,首先实例化一个queue对象,然著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshuangnet】后往这个queue push异步操作,异本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。步操作被装在一个函数中被push进队列,本文版权归作者所有,未经授权不得转载。【本文首发于唐霜的博客】它的回调函数一定是按照push的顺序执行【未经授权禁止转载】著作权归作者所有,禁止商业用途转载。

未经授权,禁止复制转载。原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net
import DefererQueue from 'deferer-queue'

const queue = new DefererQueue()

const defer1 = () => new Promise((resolve, reject) => { ... })
const defer2 = () => axios.get(...)
const defer3 = async () => ...

queue.push(defer1).then(() => { console.log(1) })
queue.push(defer2).then(() => { console.log(2) })
queue.push(defer3).then(() => { console.log(3) })

无论defer1-3中的谁,执行后谁先返本文作者:唐霜,转载请注明出处。【版权所有,侵权必究】回结果,控制台输出的结果永远是1 2 3未经授权,禁止复制转载。未经授权,禁止复制转载。(defer都成功的前提下)。

【本文受版权保护】【版权所有】唐霜 www.tangshuang.net未经授权,禁止复制转载。【原创不易,请尊重版权】

要使用上述的四种模式,只需要在实例化的时【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。候,传入一个options对象,将mod著作权归作者所有,禁止商业用途转载。【作者:唐霜】e设置为parallel/serial/【本文受版权保护】本文版权归作者所有,未经授权不得转载。switch/shift中的一个即可:

著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。
const queue = new DefererQueue({
  mode: 'switch',
})

其他的用法一样。这样,你的队列就会按照“【转载请注明来源】【作者:唐霜】尾调”的模式运行。

【关注微信公众号:wwwtangshuangnet】本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。【本文首发于唐霜的博客】【版权所有,侵权必究】

关于具体的API使用细则,你可以阅读它的文档【版权所有,侵权必究】。其中,利用axios的cancel能力【访问 www.tangshuang.net 获取更多精彩内容】转载请注明出处:www.tangshuang.net那个地方非常有借鉴意义。

【原创不易,请尊重版权】【关注微信公众号:wwwtangshuangnet】【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】【本文首发于唐霜的博客】