文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

JS前端并发多个相同的请求怎么控制为只发一个请求

2023-07-02 17:14

关注

这篇文章主要讲解了“JS前端并发多个相同的请求怎么控制为只发一个请求”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JS前端并发多个相同的请求怎么控制为只发一个请求”吧!

描述如下

并发: 一个接口请求还处于pending,短时间内就发送相同的请求

async function fetchData (a)  {    const data = await fetch('//127.0.0.1:3000/test')    const d = await data.json();    console.log(d);    return d;}fetchData(2) // 编号 1fetchData(2) // 2fetchData(2) // 3fetchData(2) // 4fetchData(2) // 4fetchData(2) // 5fetchData(2)fetchData(2)

老版本cachedAsync

我之前使用过vue的缓存函数缓存成功的请求, 实现是这样的。下面的cachedAsync只会缓存成功的请求,如果失败了,直接拉起新的请求。但是如果是上面的并发场景,相同的请求因为无法命中缓存,会出现连续发送三个请求的问题,无法处理这种并发的场景。

const cachedAsync = function(fn) {    const cache = Object.create(null);    return async str => {        const hit = cache[str];        if (hit) {            return hit;        }        // 只缓存成功的Promise, 失败直接重新请求        return (cache[str] = await fn(str));    };};const fetch3 = cachedAsync(fetchData)fetch3(2);fetch3(2);fetch3(2);

进阶版本

首先缓存是必须的,那么我们只要处理怎么控制并发即可。可以有这么一个思路

  const cacheAsync = (promiseGenerator, symbol) => {    const cache = new Map();    const never = Symbol();    return async (params) => {      return new Promise((resolve, reject) => {      // 可以提供键值        symbol = symbol || params;        let cacheCfg = cache.get(symbol);        if (!cacheCfg) {          cacheCfg = {            hit: never,            exector: [{ resolve, reject }],          };          cache.set(symbol, cacheCfg);        } else {          // 命中缓存          if (cacheCfg.hit !== never) {            return resolve(cacheCfg.hit)          }          cacheCfg.exector.push({ resolve, reject });        }        const { exector } = cacheCfg;        // 处理并发,在请求还处于pending过程中就发起了相同的请求        // 拿第一个请求        if (exector.length === 1) {          const next = async () => {            try {              if (!exector.length) return;              const response = await promiseGenerator(params);              // 如果成功了,那么直接resolve掉剩余同样的请求              while (exector.length) { // 清空                exector.shift().resolve(response);               }              // 缓存结果              cacheCfg.hit = response;            } catch (error) {              // 如果失败了 那么这个promise的则为reject              const { reject } = exector.shift();              reject(error);              next(); // 失败重试,降级为串行            }          };          next();        }      });    };  };

测试cacheAsync

需要测试的场景

快速搭建一个服务器

const koa = require("koa");const app = new koa();function sleep(seconds) { return new Promise((resolve, reject) => {   setTimeout(resolve, seconds); });}app.use(async (ctx, next) => { if (ctx.url === "/test") {   await sleep(200);   const n = Math.random();   // 随机挂掉接口   if (n > 0.8) {       ctx.body = n;   } else {       ctx.status = 404       ctx.body = ''   }   next(); }});app.listen(3000, "127.0.0.1", () => console.log("listening on 127.0.0.1:3000"));

客户端

  var fetch3 = cacheAsync(fetchData, "test2");  async function fetchData(a) {    const data = await fetch("//127.0.0.1:3000/test");    const d = await data.json();    console.log(d);    return d;  }   // 并发6个相同的请求  console.log(fetch3(2));  console.log(fetch3(2));  console.log(fetch3(2));  console.log(fetch3(2));  console.log(fetch3(2));  console.log(fetch3(2));

看下测试结果,刷新下页面

第一次运气很好,第一次接口就请求成功,只发送了一个请求

JS前端并发多个相同的请求怎么控制为只发一个请求

第二次测试运气不好,最后一个请求才成功,也是最差的场景

JS前端并发多个相同的请求怎么控制为只发一个请求

第三次测试,请求第三次成功了

JS前端并发多个相同的请求怎么控制为只发一个请求

测试下缓存 在控制台主动请求fetch3,成功命中。

JS前端并发多个相同的请求怎么控制为只发一个请求

从测试结果来看是正确的,符合了并发和缓存的场景。有人会问为什么要缓存接口,举个场景。输入关键字搜索,监听的是input事件,在你增删关键字的时候,就会出现请求参数一样的场景,这时候就符合防抖+前端接口缓存的方式。遇到相同关键字直接拉之前的缓存。

提示

这个缓存因为是闭包的方式,因此刷新页面缓存也失效了。不过我认为这个是理应如此,因为大部分场景刷新页面,就是要重置状态,如果要持久化,还不如保存到本地存储。

感谢各位的阅读,以上就是“JS前端并发多个相同的请求怎么控制为只发一个请求”的内容了,经过本文的学习后,相信大家对JS前端并发多个相同的请求怎么控制为只发一个请求这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯