【问题标题】:Improve performance for this method to flatten arrays提高此方法的性能以展平阵列
【发布时间】:2018-04-18 15:28:39
【问题描述】:

我正在开发一个大型 javascript 应用程序,不出所料,它在 IE11 中确实很困难(Chrome = 8 秒,nodejs= 8 秒,IE11 = 35 秒)。

所以我做了一些分析,发现这个方法是我的时间槽。我已经完成了我能想到的所有更改 - 是否还有其他可以改进的性能改进?

const flatten = function(arr, result) {
  if (!Array.isArray(arr)) {
    return [arr];
  }

  if(!result){
    result = [];
  }

  for (let i = 0, length = arr.length; i < length; i++) {
      const value = arr[i];
      if (Array.isArray(value)) {
        flatten(value, result);
      } 
      else {
        result.push(value);
      }
  }

  return result;
  };

该方法被多次调用,数组较小(最多 10 个字符串项,深度不超过 2 级)。

【问题讨论】:

  • 一些 JavaScript 引擎的 array.push(n)array[array.length] = n 快,其他的则相反。您可能会检查哪个适用于 IE,然后使用它更快找到的那个。
  • @T.J.Crowder - 感谢您的评论。 push() 在所有这些(IE、Chrome 和 nodejs)中似乎都更快......
  • 好的。 (Chrome 和 Node.js 相同并不奇怪——它们使用相同的 JavaScript 引擎 V8。)

标签: javascript algorithm performance internet-explorer


【解决方案1】:

应避免重复执行if (!result)Array.isArray(value) 检查。我会去的

function flatten(arr, result = []) {
  if (Array.isArray(arr)) {
    for (var i = 0; i < arr.length; i++) {
      flatten(arr[i], result);
    }
  } else {
    result.push(arr);
  }
  return result;
}

为了简单起见,如果编译器没有通过内联和识别循环模式来优化这一点,我也会尝试

function flatten(val) {
  if (Array.isArray(val)) // omit this check if you know that `flatten` is called with arrays only
    return flattenOnto(val, []);
  else
    return [val];
}
function flattenOnto(arr, result) {
  for (var i = 0, len = arr.length; i < len; i++) {
    var val = arr[i];
    if (Array.isArray(val))
      flattenOnto(val, result);
    else
      result.push(val);
  }
  return result;
}

我也使用普通的var 而不是let,因为it had been known to be faster,不知道现在是否改变了。

如您所说,如果您也知道数组的深度有限,您甚至可能想尝试内联递归调用并将其拼写为

function flatten(val) {
  if (!Array.isArray(val)) return [val]; // omit this check if you can
  var result = [];
  for (var i = 0, ilen = arr.length; i < ilen; i++) {
    var val = arr[i];
    if (Array.isArray(val)) {
      for (var j = 0, jlen = val.length; j < jlen; j++) {
        // as deep as you need it
        result.push(val[j]);
      }
    } else {
      result.push(val);
    }
  }
  return result;
}

【讨论】:

  • 如果试图勉强发挥最后一点性能,可能值得将 Array.isArray 缓存到本地。
  • @Bergi:是的,doesn't seem to make much difference(尽管它在 IE11 上确实有一点不同,这对于 OP 来说是个问题,所以......)。
  • Bergi + @T.J.Crowder - 谢谢。你的建议快 23%:jsperf.com/flattenvflatten
  • @Bergi - 是的,不幸的是,在应用程序中我没有看到太多的性能提升......但它还是好一点......
【解决方案2】:

你使用递归的方式对我来说有点奇怪:你既要返回数组又要根据深度级别改变参数。您还重复了Array.isArray(array) 呼叫。我认为这段代码可以非常简化,例如如下(你可以看到没有参数突变):

const flatten = (array) => Array.isArray(array)
  ? array.reduce((accumulated, value) => accumulated.concat(flatten(value)), [])
  : [array];

说实话,不确定性能会不会有那么大的提升,但在我看来它看起来更优雅 - jsPerf 是你的朋友!

【讨论】:

  • 使用concat 创建和丢弃大量中间数组不会提高时间性能。
  • @sp00m - 我给了你一个 +1 来帮助你,但是 jsPerf 也是你的朋友......这真的是在扼杀性能......
  • @LeonardoSeccia 是的,好吧,只是想与您分享这个工具,以防您想对收到的解决方案进行基准测试;)
猜你喜欢
  • 2020-04-21
  • 2014-11-20
  • 1970-01-01
  • 2021-04-28
  • 1970-01-01
  • 2017-11-06
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多