翼度科技»论坛 编程开发 JavaScript 查看内容

JavaScript 实现异步任务循环顺序执行

11

主题

11

帖子

33

积分

新手上路

Rank: 1

积分
33
JavaScript 实现异步任务循环顺序执行

需求场景:数组的元素作为异步任务的参数,循环遍历该数组,并执行异步任务。
一、错误的实现

简单的错误实现
  1. // 异步任务的参数数组
  2. const arr = [1, 2, 3, 4];
  3. // 异步任务函数
  4. function task(params, callback) {
  5.   setTimeout(() => {
  6.     if (!!callback) {
  7.       callback(params);
  8.     }
  9.   }, 1000);
  10. }
  11. // 循环遍历异步任务的参数数组,并执行异步任务
  12. console.time("Test code");
  13. arr.forEach((item, index) => {
  14.   task(item, (ret) => {
  15.     console.log("ret", ret);
  16.     console.timeEnd("Test code");
  17.     if (index + 1 < arr.length) {
  18.       console.time("Test code");
  19.     }
  20.   });
  21. });
复制代码
执行结果输出:

由上图可知,只有在执行第一次异步任务时耗时近 1 秒,其它异步任务执行时间远远达不到 1 秒。导致该问题的原因有待进一步深入研究。
使用 Promise.all 的错误实现
  1. // 异步任务的参数数组
  2. const arr = [1, 2, 3, 4];
  3. // 异步任务函数
  4. function task(params, callback) {
  5.   setTimeout(() => {
  6.     if (!!callback) {
  7.       callback(params);
  8.     }
  9.   }, 1000);
  10. }
  11. const tasks = [];
  12. // 循环遍历异步任务的参数数组,并执行异步任务
  13. console.time("Test code");
  14. arr.forEach((item, index) => {
  15.   tasks.push(
  16.     new Promise((resolve) => {
  17.       task(item, (ret) => {
  18.         console.log("ret", ret);
  19.         console.timeEnd("Test code");
  20.         if (index + 1 < arr.length) {
  21.           console.time("Test code");
  22.         }
  23.         resolve(ret);
  24.       });
  25.     })
  26.   );
  27. });
  28. Promise.all(tasks)
  29.   .then((values) => {
  30.     console.log(values);
  31.   })
  32.   .catch((error) => {
  33.     console.error(error);
  34.   });
复制代码
执行结果输出:

由上图可知,该种实现方式的结果和第一种方式差不多,也是只有在执行第一次异步任务时耗时 1 秒左右,剩下的异步任务执行时间达不到 1 秒。同样,该问题也需要对其深层的原因进行研究。
二、正确的实现
  1. // 异步任务的参数数组
  2. const arr = [1, 2, 3, 4];
  3. // 异步任务函数
  4. function task(params, callback) {
  5.   setTimeout(() => {
  6.     if (!!callback) {
  7.       callback(params);
  8.     }
  9.   }, 1000);
  10. }
  11. const tasks = [];
  12. console.time("Test code");
  13. arr.forEach((item, index) => {
  14.   tasks.push(function () {
  15.     return new Promise((resolve) => {
  16.       task(item, (ret) => {
  17.         console.log("ret", ret);
  18.         console.timeEnd("Test code");
  19.         if (index + 1 < arr.length) {
  20.           console.time("Test code");
  21.         }
  22.         resolve(ret);
  23.       });
  24.     });
  25.   });
  26. });
  27. // 定义一个递归函数来依次执行任务
  28. function runTasks(index) {
  29.   if (index >= tasks.length) {
  30.     // 如果所有任务都已经执行完毕,返回一个 resolved 的 Promise
  31.     return Promise.resolve();
  32.   }
  33.   // 执行当前任务,然后递归执行下一个任务
  34.   return tasks[index]().then(function () {
  35.     return runTasks(index + 1);
  36.   });
  37. }
  38. // 调用递归函数来执行任务
  39. runTasks(0)
  40.   .then(function () {
  41.     console.log("All tasks are done!");
  42.   })
  43.   .catch(function (error) {
  44.     console.error(error);
  45.   });
复制代码
执行结果输出:


来源:https://www.cnblogs.com/yuzhihui/p/17222521.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具