V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
• 请不要在回答技术问题时复制粘贴 AI 生成的内容
iugo
V2EX  ›  程序员

为什么这段代码这么慢? 使用 Array.proto.reduce 的时候, 一定要复用 accumulator ?

  •  1
     
  •   iugo ·
    iugo · 2018-06-28 11:38:48 +08:00 · 2373 次点击
    这是一个创建于 2372 天前的主题,其中的信息可能已经有所发展或是发生改变。
    第 1 条附言  ·  2018-06-28 17:36:06 +08:00
    我觉得不是声明变量再丢弃不会这么耗费时间. 可能是我不了解累加器.

    参看: https://jsbin.com/jifamucama/1/edit?js,output
    第 2 条附言  ·  2018-06-28 17:45:25 +08:00
    第 3 条附言  ·  2018-07-10 15:53:21 +08:00
    关于使用 reduce 的时候是否要修改 acc 问题, 我想这个 issue 是值得参考的: https://github.com/eslint/eslint/issues/8581
    14 条回复    2018-07-10 15:53:02 +08:00
    autoxbc
        1
    autoxbc  
       2018-06-28 12:08:35 +08:00 via iPhone
    是的,意识到这个后,我把自己代码里所有的 reduce 都改写了,尤其是展开再合并那种
    VDimos
        2
    VDimos  
       2018-06-28 12:14:42 +08:00 via Android
    不推荐使用 reduce
    Torpedo
        3
    Torpedo  
       2018-06-28 12:36:29 +08:00   ❤️ 1
    test1,test1.2,test1.3 到后来每次循环都是把一个 成百上千的对象的属性赋值给一个新的对象,能不慢么。。。
    qdwang
        4
    qdwang  
       2018-06-28 12:43:35 +08:00
    reduce 实力背锅
    jimliang
        5
    jimliang  
       2018-06-28 13:11:43 +08:00
    老铁,你用法错了,无端端生成很多没用的对象
    正确用法:
    ```
    console.time('test3');
    const dataMap33 = arr.reduce((acc, v) => {
    acc[v.id] = v.name;
    return acc;
    }, Object.create(null));
    console.timeEnd('test3');
    ```
    UnluckyNinja
        6
    UnluckyNinja  
       2018-06-28 17:09:41 +08:00
    吃饭的时候,
    吃一口,把筷子洗了,洗双新筷子,然后吃下一口,再洗掉,再换新的,
    感受一下
    iugo
        7
    iugo  
    OP
       2018-06-28 17:27:55 +08:00
    @jimliang 我一般不会直接修改函数参数, 虽然这种风险是可控的. 可能 accumulator 是个例外.
    Pastsong
        8
    Pastsong  
       2018-06-28 17:31:51 +08:00
    因为 reduce 是 immutable 的
    iugo
        9
    iugo  
    OP
       2018-06-28 17:38:44 +08:00
    auroraccc
        10
    auroraccc  
       2018-06-28 17:58:13 +08:00
    不是 reduce 的锅吧 , 主要是 ... 耗时
    iugo
        11
    iugo  
    OP
       2018-06-28 18:00:23 +08:00
    声明对象不可怕, 声明大对象是有点吓人.

    对于标题, 我觉得结论可能是:

    使用 Array.proto.reduce 的时候, 一定要复用 accumulator.
    UnluckyNinja
        12
    UnluckyNinja  
       2018-06-28 18:09:13 +08:00
    @iugo #9 testFn 根本就没用到 testAcc ……只是每次循环对其赋值,只是执行了 1000 对长度为 1 的对象添加了一个属性而已……(你添加属性的方法是返回新变量)

    你之前那个脚本问题在于对对象,你循环 n 次,每次都要对第 n-1 次循环的返回对象打散然后重组,相当于操作 n-1 次 Object.assign。也就是复杂度大约为 O(n^2),当然慢啦……
    morethansean
        13
    morethansean  
       2018-06-28 18:58:26 +08:00
    这么多写操作……
    而且光是 GC 的时间就比你第二种操作耗时多好多倍了……
    iugo
        14
    iugo  
    OP
       2018-07-10 15:53:02 +08:00
    关于使用 reduce 的时候是否要修改 acc 问题, 我想这个 issue 是值得参考的: https://github.com/eslint/eslint/issues/8581
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1123 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 25ms · UTC 18:48 · PVG 02:48 · LAX 10:48 · JFK 13:48
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.