数组扁平化的8种方法

编程入门 行业动态 更新时间:2024-10-09 15:22:00

<a href=https://www.elefans.com/category/jswz/34/1771288.html style=数组扁平化的8种方法"/>

数组扁平化的8种方法

前段时间看到一篇关于数组扁平化的公众号文章,仔细理解把几个方法试了一下之后感觉受益良多,在此基础上我又查询了其他几种方法,归纳整理后决定写下这篇文章。

什么是“数组扁平化”

用简单的话来说,就是将一个多维数组变为一个一维数组。例如,将数组[1, 2, [3, [4, 5]], [6, 7]]扁平化处理后输出[1, 2, 3, 4, 5, 6, 7]。

实现“数组扁平化”方法

方式1:使用基础的递归遍历

声明一个函数,遍历数组的每个元素,判断当前元素是否仍是数组,是的话递归执行这个函数,并把执行结果与当前结果数组合并,不是数组则直接将当前元素push到结果数组中。

function flatten(arr) {let result = [];// 此处也可使用for...of遍历for (let i = 0; i < arr.length; i++) {if (Array.isArray(arr[i])) {result = result.concat(flatten(arr[i]));} else {result.push(arr[i]);}}return result;
}

方式2:使用reduce函数递归遍历

与第一种方法思路是一样的,区别在于这里利用reduce函数的特性遍历数组并保存每一次的计算结果。

function flatten(arr) {return arr.reduce((pre, current) => {if (Array.isArray(current)) {return pre.concat(flatten(current));} else {return pre.concat(current);}}, []);
},

方式3:数组强制类型转换

对于数组对象,toString方法连接数组并返回一个字符串,其中包含用逗号分隔的每个数组元素。返回的字符串使用split分割成子字符串数组,最后将数组中每个元素的类型转换为Number型。

function flatten(arr) {return arr.toString().split(',').map((item) => Number(item));
},

为了方便理解,在浏览器调试工具的Console下逐步执行各个步骤,每步的输出结果如下图所示。

方式4:while循环结合findIndex与concat

while循环中,使用findIndex判断当前数组是否是一个多维数组(即判断数组是否存在Array类型的元素),如是,则使用…扩展操作符展开作为concat方法的参数进行合并(如下图)并赋值给当前数组,再执行下一次循环的条件判断,直至得到一个一维数组。

function flatten(arr) {while (arr.findIndex((item) => Array.isArray(item)) > 0) {arr = [].concat(...arr);}return arr;
}

可能有些小伙伴不太能理解 arr = [].concat(...arr);这一行代码,其实这一行代码的作用就相当于把所给多维数组剥开一层(如下图所示),多次循环执行直至得到一个一维数组。

方式5:直接使用ES6的flat方法

直接使用ES6提供的flat方法实现扁平化。falt()方法会按照指定的深度递归遍历数组,arr.flat([depth]),参数depth不填时默认值为1,depth为Infinity表示展开任意深度的嵌套数组。

function flatten(arr) {return arr.flat(Infinity);
}

方式6:使用JSON的函数和正则表达式

使用JSON的序列化函数stringify()先对数组进行序列化,再用正则去掉[],得到的结果在最外层加上[]后使用JSON.parse()恢复成数组对象。

function flatten(arr) {let result = JSON.stringify(arr).replace(/(\[|\])/g,'');result = '[' + result + ']';return JSON.parse(result);
}

方式7:使用堆栈stack

创建一个栈结构和一个存放结果的空数组,然后遍历栈结构,判断元素如果是数组就使用扩展操作符展开再次入栈,不是就添加到结果数组的开头。

// 无递归数组扁平化
function flatten(arr) {const stack = [...arr];const res = [];while (stack.length) {// 出栈 从 stack 中取出一个元素const next = stack.pop();if (Array.isArray(next)) {// 展开一层重新入栈stack.push(...next);} else {res.unshift(next);}}return res;
}

方式8:使用Generator 函数与递归结合

Generator 函数是 ES6 提供的一种异步编程解决方案,语法行为与传统函数完全不同。思路还是遍历当前数组,判断数组元素是否是数组,是则使用 yield 语句递归执行这个 Generator 函数,不是则使用 yield 表达式返回当前值。

function* flatten(arr){for(const item of arr) {if(Array.isArray(item)) {yield* flatten(item);} else {yield item;}}
}
const array = [1, 2, [3, [4, 5]], [6, 7]];
const flattened = [...flatten(array)];

看完这8种方式,是否发现还是ES6提供的方法好用啊,还能指定扁平化的深度,其实以上其它几种方法也是可以修改成支持层数扁平化的,具体可以参考MDN文档flat()方法的替代方案。

更多推荐

数组扁平化的8种方法

本文发布于:2024-02-28 00:00:16,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1766985.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:数组   种方法   扁平化

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!