• 为什么 array.foreach 不支持 async/await


    一、背景


    react 项目中,渲染组件时,显示的数据一直有问题,本来以为是 react 组件的问题,后来才发现罪魁祸首在 fetch 数据的过程,因为我用了 async/await ,而却搭配了 foreach 去循环拉取数据,却导致本以为是同步的操作还是变成了异步。

    二、正文


    沿用我之前一篇文章(callback vs async.js vs promise vs async / await)里的例子,来重现这个错误:

    let read = function (code) {
       if (code) {
           return true;
       } else {
           return false;
       }
    }
    
    let readFileA = function () {
       return new Promise(function (resolve, reject) {
           if (read(1)) {
               resolve("111");
           } else {
               reject("a fail");
           }
       });
    }
    let readFileB = function () {
       return new Promise(function (resolve, reject) {
           if (read(1)) {
               resolve("222");
           } else {
               reject("b fail");
           }
       });
    }
    let readFileC = function () {
       return new Promise(function (resolve, reject) {
           if (read(1)) {
               resolve("333");
           } else {
               reject("c fail");
           }
       });
    }
    
    async function test() {
       try {
    
           let readFileFun = [readFileA(), readFileB(), readFileC()]
    
           console.log("………………start………………")
    
           // // 方法一:forEach
           // await readFileFun.forEach(async (func, i) => {
           //     console.log("start:", i+1)
           //     let re = await func;
           //     console.log(re)
           //     console.log("end:", i+1)
           // }) 
    
           // // 方法二:for loop
           // for (let i = 0; i < readFileFun.length; ++i) {
           //     console.log("start:", i+1)
           //     let re = await readFileFun[i];
           //     console.log(re)
           //     console.log("end:", i+1)
           // }
    
           // // 方法三:for ... of
           // for (const [i, func] of readFileFun.entries()) {
           //     console.log("start:", i+1)
           //     let re = await func;
           //     console.log(re)
           //     console.log("end:", i+1)
           // }
    
           console.log("………………end………………")
    
       } catch (err) {
           console.log(err); // 如果b失败,return: b fail
       }
    }
    
    test();  
    

    输出结果:

    # (错)方法一:
    ………………start………………
    start: 1
    start: 2
    start: 3
    111
    end: 1
    222
    end: 2
    333
    end: 3
    ………………end………………
    
    
    # (对)方法二、三:
    ………………start………………
    start: 1
    111
    end: 1
    start: 2
    222
    end: 2
    start: 3
    333
    end: 3
    ………………end………………
    

    为什么 foreach 不行,而 普通 for 循环 和 for…of 却正常呢?

    我们得先从 foreach 的源码看起:(https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach>)

    // Production steps of ECMA-262, Edition 5, 15.4.4.18
    // Reference: http://es5.github.io/#x15.4.4.18
    if (!Array.prototype.forEach) {
    
      Array.prototype.forEach = function(callback/*, thisArg*/) {
    
        var T, k;
    
        if (this == null) {
          throw new TypeError('this is null or not defined');
        }
    
        // 1. Let O be the result of calling toObject() passing the
        // |this| value as the argument.
        var O = Object(this);
    
        // 2. Let lenValue be the result of calling the Get() internal
        // method of O with the argument "length".
        // 3. Let len be toUint32(lenValue).
        var len = O.length >>> 0;
    
        // 4. If isCallable(callback) is false, throw a TypeError exception. 
        // See: http://es5.github.com/#x9.11
        if (typeof callback !== 'function') {
          throw new TypeError(callback + ' is not a function');
        }
    
        // 5. If thisArg was supplied, let T be thisArg; else let
        // T be undefined.
        if (arguments.length > 1) {
          T = arguments[1];
        }
    
        // 6. Let k be 0.
        k = 0;
    
        // 7. Repeat while k < len.
        while (k < len) {
    
          var kValue;
    
          // a. Let Pk be ToString(k).
          //    This is implicit for LHS operands of the in operator.
          // b. Let kPresent be the result of calling the HasProperty
          //    internal method of O with argument Pk.
          //    This step can be combined with c.
          // c. If kPresent is true, then
          if (k in O) {
    
            // i. Let kValue be the result of calling the Get internal
            // method of O with argument Pk.
            kValue = O[k];
    
            // ii. Call the Call internal method of callback with T as
            // the this value and argument list containing kValue, k, and O.
            callback.call(T, kValue, k, O);
          }
          // d. Increase k by 1.
          k++;
        }
        // 8. return undefined.
      };
    }
    

    摘抄最重要的部分:

    
    /* 
    O 为传入数组
    len 为传入数组长度
    callback 为传入回调函数
    */
    
    while (k < len) {
    
          var kValue; 
          if (k in O) { 
            kValue = O[k]; 
            callback.call(T, kValue, k, O);
          } 
      
          k++;
    }
    

    可以看到callback.call(T, kValue, k, O);这一句,callback 其实是我们传入的一个被 async 封装的 promise 对象,而 Array.prototype.forEach 内部并未对这个promise 对象做任何处理,只是忽略它。

    如果我们尝试把 Array.prototype.forEach 改造一下,让它不要忽视,就可以达到效果了,如下:

     Array.prototype.forEach = async function(callback/*, thisArg*/) {
       
       		// ………
    			await callback.call(T, kValue, k, O);
       		// ………
             
    	};
    

    解决方案

    你总不能去侵入式的改造Array.prototype.forEach吧!所以最简单的办法就是抛弃 foreach,使用 for…of 或者 for 循环!


    参考资料

    https://stackoverflow.com/questions/37576685/using-async-await-with-a-foreach-loop

    https://github.com/babel/babel/issues/909

  • 相关阅读:
    Git中使用.gitignore忽略文件的推送
    git stash详解
    Git撤销&回滚操作(git reset 和 get revert)
    git rebase和git merge的区别
    git撤销已经push到远程仓库上的代码
    Git Merge
    git cherry-pick 教程
    Failed to start LSB: Bring up/down错误解决方法
    linux centos7安装部署gitlab服务器
    CentOs7 HP找回root密码
  • 原文地址:https://www.cnblogs.com/xjnotxj/p/10629900.html
Copyright © 2020-2023  润新知