JavaScript进行数组扁平化的四种方法详解与最佳实践

 更新时间:2025年11月13日 09:58:56   作者:木易 士心  
在日常开发中,我们经常会遇到嵌套数组的处理需求,本文将系统性地介绍四种主流的数组扁平化方法并从多个维度进行对比分析,希望对大家有所帮助

在日常开发中,我们经常会遇到嵌套数组(nested arrays)的处理需求。例如,从 API 接口返回的数据可能包含多层结构,而我们需要将其“拍平”成一维数组以便后续操作。本文将以如下示例数据为切入点,系统性地介绍四种主流的数组扁平化方法,并从语法简洁性、执行性能、兼容性、适用场景等多个维度进行对比分析。

const arr = [[1, 2], [3, 4], [5, 6]];
// 目标:[1, 2, 3, 4, 5, 6]

方法 1:Array.prototype.flat()(现代标准,推荐)

基本用法

ES2019(ECMAScript 2019)引入了 flat() 方法,专门用于数组扁平化,语法简洁直观:

const merged = arr.flat();
console.log(merged); // [1, 2, 3, 4, 5, 6]

深度控制

默认只展开一层嵌套。

可通过传入参数指定展开深度:

const deepArr = [[[1, 2]], [[3, 4]]];
console.log(deepArr.flat(2));        // [[1, 2], [3, 4]]
console.log(deepArr.flat(Infinity)); // [1, 2, 3, 4] — 完全扁平化

优势

  • 语义清晰:专为扁平化设计,意图明确。
  • 性能良好:原生实现,V8 等引擎高度优化。
  • 不可变性:不修改原数组,返回新数组。

注意事项

  • 不支持 IE 和部分旧版移动端浏览器(需 Babel 转译或 polyfill)。
  • 对于仅需处理单层嵌套的场景,flat()首选方案

方法 2:扩展运算符 +concat()

实现方式

利用 ES6 的扩展运算符(spread operator)将子数组作为参数传入 concat

const merged = [].concat(...arr);
console.log(merged); // [1, 2, 3, 4, 5, 6]

原理剖析

  • ...arr[[1,2], [3,4], [5,6]] 展开为三个独立参数:[1,2], [3,4], [5,6]
  • [].concat(a, b, c) 等价于合并这些数组。

优缺点

优点

  • 代码简短,一行搞定。
  • 兼容 ES6+ 环境(现代浏览器和 Node.js 广泛支持)。

缺点

  • 存在栈溢出风险:当子数组数量极大(如 > 10⁵)时,函数调用栈可能超出限制(因 concat 接收的是展开后的参数列表)。
  • 仅适用于单层嵌套,无法处理深层结构。

适用建议:适合中小型数组、追求代码简洁性的场景。

方法 3:Array.prototype.reduce()

经典写法

const merged = arr.reduce((acc, subArr) => acc.concat(subArr), []);

性能优化写法(使用扩展运算符)

const merged = arr.reduce((acc, subArr) => [...acc, ...subArr], []);

深度解析

  • 第一种写法:每次调用 concat 都会创建一个新数组,时间复杂度为 O(n²),大数据量下性能较差。
  • 第二种写法:虽然避免了 concat,但 [...acc, ...subArr] 同样会复制整个累加器数组,内存开销大。

适用场景

需要兼容 ES5 或更老环境(如 IE11)。

需要在扁平化过程中加入自定义逻辑(如过滤、映射等):

arr.reduce((acc, sub) => {
  return acc.concat(sub.filter(x => x > 2));
}, []);

总结

  • 灵活性高,但性能较低
  • 除非有特殊需求,否则不建议在现代项目中作为首选。

方法 4:for...of循环 +push(...subArr)

高性能实现

const merged = [];
for (const subArr of arr) {
  merged.push(...subArr);
}
console.log(merged); // [1, 2, 3, 4, 5, 6]

为什么高效

  • 原地操作:直接向结果数组追加元素,避免中间数组创建。
  • 无函数调用开销:相比 reduceconcat,循环开销极小。
  • 内存友好:适合处理超大数组(如百万级元素)。

注意事项

仍受限于扩展运算符的参数数量上限(理论上 V8 引擎限制约为 65535 个参数),若单个子数组极大(如长度 > 10⁵),可改用 push.apply 或分批处理:

for (const subArr of arr) {
  Array.prototype.push.apply(merged, subArr);
}

push.apply 在极端情况下也可能栈溢出,最稳妥的方式是使用普通 for 循环逐个 push

适用场景

  • 高性能要求(如数据处理、游戏引擎、实时计算)。
  • 处理大型或不确定规模的嵌套数组。

综合对比表

方法语法简洁性执行性能内存效率兼容性适用嵌套深度是否修改原数组
arr.flat()⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐ES2019+可控(默认1层)
[].concat(...arr)⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐ES6+仅1层
reduce() + concat⭐⭐⭐广泛(ES5+)仅1层
for...of + push(...)⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐ES6+仅1层

注:⭐ 越多表示越优。

最佳实践建议

现代项目(支持 ES2019+):优先使用 arr.flat() —— 简洁、安全、高效。

需要兼容旧环境(如 IE11):使用 reduce 或手动 for 循环,并搭配 Babel 转译。

处理超大数组(> 10⁵ 元素):选择 for...of + push,并注意单个子数组长度限制。

需要在扁平化时附加逻辑(如去重、过滤):使用 reduce,灵活组合业务逻辑。

延伸思考:完全扁平化任意深度嵌套

若面对如下结构:

const deeplyNested = [1, [2, [3, [4, [5]]]]];

可封装递归函数或使用 flat(Infinity)

const flatten = arr => arr.flat(Infinity);
// 或
const flatten = arr => arr.reduce(
  (acc, val) => acc.concat(Array.isArray(val) ? flatten(val) : val),
  []
);

提示:flat(Infinity) 在大多数现代引擎中已高度优化,通常优于手写递归。

结语

数组扁平化虽是一个“小问题”,却能反映出开发者对语言特性、性能边界和工程实践的理解深度。在实际项目中,应根据运行环境、数据规模、维护成本综合选择方案。对于绝大多数现代前端或 Node.js 应用,arr.flat() 已足够优雅且高效——简单,才是终极的复杂。

到此这篇关于JavaScript进行数组扁平化的四种方法详解与最佳实践的文章就介绍到这了,更多相关JavaScript数组扁平化内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js canvas实现5张图片合成一张图片

    js canvas实现5张图片合成一张图片

    这篇文章主要为大家详细介绍了js canvas实现5张图片合成一张图片,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-07-07
  • js统计录入文本框中字符的个数并加以限制不超过多少

    js统计录入文本框中字符的个数并加以限制不超过多少

    为了更直观的体现用户在文本框输入文本时能看到自己输入了多少字,并且有些特殊的要求字数不超过多少,本文给出了具体的实现
    2014-05-05
  • uniapp添加操作日志的方法(uniapp、日志、文件、html5+)

    uniapp添加操作日志的方法(uniapp、日志、文件、html5+)

    近期一直在写微信小程序,有一个问题一直没有解决,就是在测试环境中调试代码会打印很多日志,方便看到问题所在,这篇文章主要给大家介绍了关于uniapp添加操作日志(uniapp、日志、文件、html5+)的相关资料,需要的朋友可以参考下
    2023-11-11
  • 解决layer图标icon不加载的问题

    解决layer图标icon不加载的问题

    今天小编就为大家分享一篇解决layer图标icon不加载的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-09-09
  • 前端必会的图片懒加载(三种方式)

    前端必会的图片懒加载(三种方式)

    在我们访问一个图片展示比较多的网页时,加载速度慢很多时候正是因为图片多导致,本文主要介绍了前端必会的图片懒加载(三种方式),具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • JS如何循环遍历JSON数据

    JS如何循环遍历JSON数据

    这篇文章主要介绍了JS如何循环遍历JSON数据的方法,本文提供了 JS 循环 JSON 数据列,以及 JS 循环遍历 JSON 数据的例子,需要的朋友可以参考下
    2024-01-01
  • javascript 45种缓动效果 非常酷

    javascript 45种缓动效果 非常酷

    javascript 45种缓动效果 非常酷,喜欢的朋友运行下看下效果。
    2011-06-06
  • js实现继承的方法及优缺点总结

    js实现继承的方法及优缺点总结

    这篇文章主要给大家介绍了关于js实现继承的方法及优缺点的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用JavaScript具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-05-05
  • js 实现在2d平面上画8的方法

    js 实现在2d平面上画8的方法

    今天小编就为大家分享一篇js 实现在2d平面上画8的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-10-10
  • JavaScript输入框字数实时统计更新

    JavaScript输入框字数实时统计更新

    这篇文章主要介绍了JavaScript输入框字数实时统计更新,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06

最新评论