数组扁平化是将多层嵌套数组转化为一层数组,常用方法包括ES2019的flat()、递归、reduce结合递归及迭代法;flat()性能好且简洁,适合现代环境,递归灵活但可能栈溢出,迭代法可避免栈溢出,适用于深度嵌套场景。

JavaScript数组扁平化,简单来说,就是把一个多层嵌套的数组(也就是二维、三维甚至更多维的数组)转换成一个只有一层的数组。核心思路无非就是遍历数组的每个元素,如果遇到还是数组的元素,就得想办法把它“展开”,直到所有元素都不再是数组为止。这在处理某些后端返回的复杂数据结构,或者需要对所有数据进行统一操作时,显得尤为重要。
要实现数组扁平化,其实有几种常见的思路,每种都有其适用场景和一些小考量。
1. 使用 Array.prototype.flat()
const nestedArray = [1, [2, 3], [4, [5, 6, [7, 8]]], 9]; // 扁平化一层 const flatOnce = nestedArray.flat(); console.log(flatOnce); // 输出: [1, 2, 3, 4, [5, 6, [7, 8]], 9] // 扁平化指定深度 const flatTwoLevels = nestedArray.flat(2); console.log(flatTwoLevels); // 输出: [1, 2, 3, 4, 5, 6, [7, 8], 9] // 扁平化所有层级(Infinity 表示任意深度) const flatAll = nestedArray.flat(Infinity); console.log(flatAll); // 输出: [1, 2, 3, 4, 5, 6, 7, 8, 9]
flat()
depth
1
Infinity
2. 递归实现: 这是理解扁平化原理最基础的方式,即便有了
flat()
function flattenDeepRecursive(arr) {
let result = [];
arr.forEach(item => {
if (Array.isArray(item)) {
// 如果是数组,递归调用自身,并将结果合并到当前结果数组
result = result.concat(flattenDeepRecursive(item));
} else {
// 如果不是数组,直接添加到结果数组
result.push(item);
}
});
return result;
}
const nestedArray = [1, [2, 3], [4, [5, 6, [7, 8]]], 9];
const flatArray = flattenDeepRecursive(nestedArray);
console.log(flatArray); // 输出: [1, 2, 3, 4, 5, 6, 7, 8, 9]这个递归版本非常直观,遇到数组就“钻进去”继续扁平化,直到遇到非数组元素才“吐出来”。
3. 使用 reduce
reduce
function flattenWithReduce(arr) {
return arr.reduce((acc, val) => {
// 如果当前值是数组,就递归扁平化它,然后和累加器合并
// 否则,直接将当前值添加到累加器
return acc.concat(Array.isArray(val) ? flattenWithReduce(val) : val);
}, []); // 初始累加器为空数组
}
const nestedArray = [1, [2, 3], [4, [5, 6, [7, 8]]], 9];
const flatArray = flattenWithReduce(nestedArray);
console.log(flatArray); // 输出: [1, 2, 3, 4, 5, 6, 7, 8, 9]这个方法写起来很简洁,对熟悉
reduce
我个人觉得,扁平化数组的需求往往源于数据结构和业务逻辑之间的“不对称”。很多时候,后端接口为了表达数据之间的层级关系,会返回多层嵌套的数据,这在概念上很清晰。但到了前端,尤其是需要展示一个列表、进行统一搜索、或者对所有叶子节点进行某种计算时,这种嵌套结构反而成了障碍。
比如,你可能从API得到一个复杂的菜单结构,里面有主菜单、子菜单、再子菜单,但现在产品经理说:“我们要做一个全局搜索,能搜到所有菜单项,不管它藏得多深。”这时候,你拿着一个多维数组去遍历查找,显然不如把它先扁平化成一个一维数组来得痛快。
再举个例子,假设你要统计一个组织架构里所有员工的工龄总和,而这个组织架构也是按部门、小组层层嵌套的。如果不对数据进行扁平化处理,你得写好几层循环才能触及到所有员工数据,而扁平化后,你只需要一次遍历就能搞定。
对我来说,扁平化更多是为了简化后续的数据处理逻辑。当数据“平”了,很多算法和操作就变得直观、线性,减少了处理层级嵌套带来的心智负担和代码复杂度。
在选择扁平化方法时,性能和适用场景是两个绕不开的话题。
Array.prototype.flat()
Infinity
递归实现的方案,包括我们自己手写的循环递归或者
reduce
flat()
那么,什么时候选哪个呢?
flat()
reduce
在实际操作中,扁平化并不是简单的“一键到底”,有些细节和挑战需要留意。
一个最典型的挑战就是前面提到的“栈溢出”。当你使用递归方法去扁平化一个嵌套层级极深的数组时,比如一个数组里套着一个数组,这个数组又套着一个数组,层层叠叠几千上万层,JavaScript引擎的调用栈是有限的,很快就会因为递归调用层数过多而耗尽,抛出
RangeError: Maximum call stack size exceeded
如何避免栈溢出? 最有效的方法就是使用迭代(非递归)的扁平化算法。这种方法通常会借助一个栈(或者队列)来模拟递归过程,但避免了实际的函数调用栈累积。
一个简单的迭代扁平化实现思路:
function flattenIterative(arr) {
const result = [];
const stack = [...arr]; // 将初始数组的所有元素放入栈中
// 当栈不为空时,持续处理
while (stack.length > 0) {
const item = stack.shift(); // 取出栈顶元素(这里用shift模拟队列,保证顺序)
if (Array.isArray(item)) {
// 如果是数组,将其所有子元素(展开后)放回栈的头部
// 这样可以确保子数组的元素在当前层级的其他元素之前被处理
stack.unshift(...item);
} else {
// 如果不是数组,就添加到结果数组
result.push(item);
}
}
return result;
}
const deepNestedArray = [1, [2, [3, [4, [5, [6, [7, [8, [9, [10, /* ... 很多层 */]]]]]]]]]]];
// 假设这里 deepNestedArray 嵌套了上千层
// const flatArray = flattenDeepRecursive(deepNestedArray); // 这可能会栈溢出
const flatArray = flattenIterative(deepNestedArray); // 这个通常不会
console.log(flatArray); // [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, ...]这种迭代方式,无论数组嵌套多深,都不会导致栈溢出,因为它只依赖于循环和数组操作,而不是函数调用栈。
其他挑战:
null
undefined
Set
Map
flat()
depth
以上就是js 怎么实现数组扁平化的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号