JS实现并发控制、请求合并、数据缓存

引言

在现代前端开发中,随着应用复杂度增加,接口请求管理成为性能优化的关键点。本文将深入探讨如何通过并发控制、数据缓存和请求合并等技术手段,构建高效、稳定的大规模请求解决方案。

并发控制

1.1 问题背景

当页面同时发起大量请求时,可能导致:

  • 浏览器TCP连接数达到上限(Chrome同一域名下默认6个)
  • 服务器压力剧增
  • 低优先级请求阻塞关键请求

1.2 解决方案

使用请求队列实现控制并发

js 复制代码
class RequestQ {
  private _running;
  private max;
  private queue: ((value: unknown) => void)[];
  constructor(max: number) {
    this.max = max;
    this.queue = [];
    this._running = 0;
  }

  async add(fn: () => Promise<any>) {
    if (this._running >= this.max) {
      await new Promise((resolve) => this.queue.push(resolve));
    }
    this._running++;

    try {
      await fn();
    } finally {
      this._running--;
      if (this.queue.length > 0) {
        this.queue.shift()!();
      }
    }
  }
}

这里主要使用Promise来控制任务的并发,当当前运行的个数大于max时,会挂起一个Promise等待它的resolve,这里是实现并发控制的关键。

请求合并

js 复制代码
class RequestMerge {
  private pending;

  constructor() {
    this.pending = new Map();
  }

  async request(key: string, fn: Function) {
    if (this.pending.get(key)) {
      return this.pending.get(key);
    }
    const promise = fn().finally(() => {
      this.pending.delete(key);
    });
    this.pending.set(key, promise);
    return promise;
  }
}

使用一个Map用于储存正在请求的网络请求,当相同请求来的时候,先在Map里去找是否有相同key的值,如果有直接返回,没有在去请求。

数据缓存

js 复制代码
class CacheRequest {
  private apiCache: Map<string, any>;

  constructor() {
    this.apiCache = new Map<string, any>();
    setInterval(() => {
      this.apiCache.forEach((value, key) => {
        if (Date.now() > value.expireTime) {
          this.apiCache.delete(key);
        }
      });
    }, 60000);
  }

  async request(key: string, fn: () => Promise<any>) {
    if (this.apiCache.has(key)) {
      return this.apiCache.get(key);
    }
    const expireTime = Date.now() + 300000;

    // eslint-disable-next-line no-useless-catch
    try {
      const result = await fn();
      this.apiCache.set(key, { result, expireTime });
      return result;
    } catch (error) {
      throw error;
    }
  }
}

类似与合并请求,不同的是这里是启动一个定时器,定期的去清理过期的key

相关推荐
凌乱风雨12113 分钟前
从源码角度解析C++20新特性如何简化线程超时取消
前端·算法·c++20
两个西柚呀6 分钟前
每日前端面试题-css塌陷
前端·css
IT_陈寒9 分钟前
Vite 5大实战优化技巧:让你的开发效率提升200%|2025前端工程化指南
前端·人工智能·后端
未来之窗软件服务18 分钟前
幽冥大陆(八十八 ) 操作系统应用封装技术C#自解压 —东方仙盟练气期
java·前端·c#·软件打包·仙盟创梦ide·东方仙盟·阿雪技术观
沛沛rh451 小时前
React 学习笔记:State、hook —— 组件的记忆
前端·javascript·react.js
0和1的舞者9 小时前
Spring AOP详解(一)
java·开发语言·前端·spring·aop·面向切面
web小白成长日记9 小时前
在Vue样式中使用JavaScript 变量(CSS 变量注入)
前端·javascript·css·vue.js
QT 小鲜肉9 小时前
【Linux命令大全】001.文件管理之which命令(实操篇)
linux·运维·服务器·前端·chrome·笔记
C_心欲无痕9 小时前
react - useImperativeHandle让子组件“暴露方法”给父组件调用
前端·javascript·react.js
BullSmall11 小时前
支持离线配置修改及删除操作的实现方案
前端