控制并发执行的解决方法(队列控制类)

前言

之前我有一篇文章(组件阅后即焚?挂载即卸载!看完你就理解了 - 掘金 (juejin.cn))提到了,工作生涯接到过一个需求是批量导出学生二维码,其中有个问题在于使用html2canvas时,会同时产生大量canvas标签占用内存导致页面卡顿甚至崩溃的问题。

思路

其实很简单,最大的原因是同时存在过多的canvas标签,占用了过多的内存,而同样作为一个热门的第三方库,html2canvas其实会主动删除自己复制的dom元素,但在并发量过的情况下,释放的速度比不上生成的速度,所以会出现问题。

那么解决方案也就是控制它的生成速度,以保障释放速度在承受范围之内。

但这里很有意思的一点是,不同电脑的承受范围是不一样的,不同的canvas占用的内存也是不一样的,所以如何控制这个量是要自己掌握的。

像一些不好掌握的情况,我的建议是直接服务端整一个服务专门用来批量执行这样的任务。

当然了,这里我们直接写个最简单易懂的方案来做就好了,控制并发量。

编程!启动!

众所周知,JS是世界上最好的语言(bushi)

hhhhh,玩个梗。

那么,为了通用,所以我们还是会以一些比较好用的方式也来设计这个东西,就决定用类class来做这个方案的载体。也就是并发队列控制类

这里我就直接放上代码,供大家参考了

js 复制代码
class ConcurrencyControllerQueue<T> {
  activeCount: number
  values: T[]

  private tasks: Array<() => Promise<T>>
  private readonly maxConcurrency: number

  constructor(maxConcurrency: number) {
    this.tasks = []
    this.values = []
    this.maxConcurrency = maxConcurrency
    this.activeCount = 0
  }

  // 将一个任务添加到队列并立即执行
  public pushTask(fn: () => Promise<T>): void {
    if (this.activeCount >= this.maxConcurrency) {
      this.tasks.push(fn)
    } else {
      this.runTask(fn)
    }
  }
  
  public clearTask(): void {
    this.tasks = []
  }

  public clearValues(): void {
    this.values = []
  }
  
  // 从队列中取出一个任务并执行
  private runTask(fn: () => Promise<T>): void {
    this.activeCount++
    fn().then((value) => {
      this.values.push(value)
      this.activeCount--
      if (this.tasks.length > 0) {
        this.runTask(this.tasks.shift()!)
      }
    })
  }
}

这个类的组成成分其实还蛮清晰的: activeCount: 当前正在活动的任务数 values: 这里我是因为学生的二维码需要以Blob数据作为任务结果,所以需要这么个东西

private tasks: 就是当前正在等待的任务队列 private readonly maxConcurrency: 最大的同时执行任务数量,这里因为觉得一个实例就单一处理一个并发量,就不让它能更改了,所以是只读

方法成员就不跟大家介绍了,方法名简单易懂

这里其实还能做更多扩展,比如设置暂停和重启、修改并发量、只是将任务添加进队列但不立即启动等等......

因为像这些扩展都并不难,且现有功能已满足了我的需求,所以就先这样,等以后有需要再修改便可。

相关推荐
云水一下1 分钟前
Vue.js从零到精通系列(三):组件化基础——Props、Emits、插槽与生命周期
前端·javascript·vue.js
小糯米60110 分钟前
JavaScript表达式与运算符
开发语言·javascript·ecmascript
SEO_juper28 分钟前
新独立站冷启动收录全攻略:配置、推送、抓取配额优化完整手册
前端·谷歌·seo·跨境电商·外贸·geo·独立站
TinssonTai32 分钟前
这个 VS Code 插件让我的 AI Coding 又快又稳 - 旧瓶装新酒
前端·人工智能·程序员
体验家33 分钟前
体验家 XMPlus 网页端问卷 SDK 技术解析:用几行 JavaScript 实现精准场景触发与防打扰机制
开发语言·前端·javascript
VidDown38 分钟前
VidDown 工具站:视频分辨率技术
javascript·网络·编辑器·音视频·视频编解码·视频
Maimai108081 小时前
Web3 前端交易系统如何落地:从下单 UI 到 Operation 编码、签名与实时状态更新
前端·react.js·ui·架构·前端框架·web3
kidding7231 小时前
高效备忘清单工具类小程序
前端·计算机网络·微信小程序·小程序
IMPYLH1 小时前
HTML 的 <abbr> 元素
前端·算法·html
小鹿软件办公1 小时前
倒计时开启:Chromium 宣布几周内将全面切断 MV2 扩展支持
开发语言·javascript·ublock origin