当前位置: 首页 > news >正文

JavaScript实现一个函数,将数组扁平化(flatten),即把多维数组转为一维数组。

大白话实现一个函数,将数组扁平化(flatten),即把多维数组转为一维数组。

思路

实现数组扁平化的基本思路是遍历数组中的每个元素,如果元素是数组,就递归地将其扁平化并添加到结果数组中;如果元素不是数组,就直接添加到结果数组中。

代码示例

function flatten(arr) {
    // 创建一个空数组,用于存储扁平化后的结果
    let flattened = [];
    // 遍历输入数组中的每个元素
    for (let i = 0; i < arr.length; i++) {
        // 检查当前元素是否为数组
        if (Array.isArray(arr[i])) {
            // 如果当前元素是数组,递归调用 flatten 函数对其进行扁平化
            // 并将扁平化后的结果添加到 flattened 数组中
            flattened = flattened.concat(flatten(arr[i]));
        } else {
            // 如果当前元素不是数组,直接将其添加到 flattened 数组中
            flattened.push(arr[i]);
        }
    }
    // 返回扁平化后的数组
    return flattened;
}

// 测试示例
const multiDimensionalArray = [1, [2, [3, 4], 5], 6, [7, 8]];
const flattenedArray = flatten(multiDimensionalArray);
console.log(flattenedArray);

代码解释

  1. 函数定义

    • function flatten(arr):定义了一个名为 flatten 的函数,它接受一个数组 arr 作为参数。
  2. 初始化结果数组

    • let flattened = [];:创建一个空数组 flattened,用于存储扁平化后的结果。
  3. 遍历数组

    • for (let i = 0; i < arr.length; i++):使用 for 循环遍历输入数组 arr 中的每个元素。
  4. 检查元素类型

    • if (Array.isArray(arr[i])):使用 Array.isArray 方法检查当前元素是否为数组。
  5. 递归处理数组元素

    • flattened = flattened.concat(flatten(arr[i]));:如果当前元素是数组,递归调用 flatten 函数对其进行扁平化,并使用 concat 方法将扁平化后的结果添加到 flattened 数组中。
  6. 处理非数组元素

    • flattened.push(arr[i]);:如果当前元素不是数组,直接使用 push 方法将其添加到 flattened 数组中。
  7. 返回结果

    • return flattened;:返回扁平化后的数组。

除了递归,还有其他方法可以实现数组扁平化吗?

除了递归方法,还有几种不同的方式可以实现数组扁平化,下面为你详细介绍:

方法一:使用 reduceconcat 方法

reduce 方法可以对数组中的每个元素执行一个提供的函数,并将其结果汇总为单个值。结合 concat 方法可以有效实现数组扁平化。

function flatten(arr) {
    // 使用 reduce 方法遍历数组
    return arr.reduce((acc, val) => {
        // 如果当前元素是数组,使用 concat 方法将其扁平化后添加到累加器中
        // 如果不是数组,直接添加到累加器中
        return acc.concat(Array.isArray(val)? flatten(val) : val);
    }, []);
}

// 测试示例
const multiDimensionalArray = [1, [2, [3, 4], 5], 6, [7, 8]];
const flattenedArray = flatten(multiDimensionalArray);
console.log(flattenedArray);

此代码中,reduce 方法遍历数组的每个元素,若元素是数组,就递归调用 flatten 函数将其扁平化,再用 concat 方法添加到累加器;若不是数组则直接添加到累加器。

方法二:使用 flat 方法

flat 是 JavaScript 数组自带的方法,能直接实现数组扁平化。可以指定扁平化的深度,若要完全扁平化,可传入 Infinity

const multiDimensionalArray = [1, [2, [3, 4], 5], 6, [7, 8]];
// 使用 flat 方法并传入 Infinity 实现完全扁平化
const flattenedArray = multiDimensionalArray.flat(Infinity);
console.log(flattenedArray);

这种方法简洁直观,直接调用 flat 方法并传入 Infinity 就能把多维数组完全扁平化为一维数组。

方法三:使用 while 循环和 some 方法

通过 some 方法检查数组中是否还有嵌套数组,若有就使用 concat 方法展开,借助 while 循环持续处理,直至没有嵌套数组。

function flatten(arr) {
    let isNested = true;
    // 当数组中还有嵌套数组时继续循环
    while (isNested) {
        isNested = false;
        let newArr = [];
        // 遍历数组
        for (let i = 0; i < arr.length; i++) {
            if (Array.isArray(arr[i])) {
                // 如果是数组,将其元素添加到新数组中
                newArr = newArr.concat(arr[i]);
                isNested = true;
            } else {
                // 不是数组则直接添加到新数组
                newArr.push(arr[i]);
            }
        }
        arr = newArr;
    }
    return arr;
}

// 测试示例
const multiDimensionalArray = [1, [2, [3, 4], 5], 6, [7, 8]];
const flattenedArray = flatten(multiDimensionalArray);
console.log(flattenedArray);

此代码利用 while 循环和 some 方法不断检查并展开嵌套数组,直到数组中不存在嵌套数组,最终得到一维数组。

哪种方法实现数组扁平化的效率更高?

递归方法

function flatten(arr) {
    let flattened = [];
    for (let i = 0; i < arr.length; i++) {
        if (Array.isArray(arr[i])) {
            flattened = flattened.concat(flatten(arr[i]));
        } else {
            flattened.push(arr[i]);
        }
    }
    return flattened;
}
  • 优点:代码逻辑直观,易于理解和实现,适合处理嵌套层级不深、数据规模较小的数组。
  • 缺点:当数组嵌套层级很深时,递归调用会导致大量的函数调用栈,可能引发栈溢出错误,而且函数调用的开销也会使性能下降。

使用 reduceconcat 方法

function flatten(arr) {
    return arr.reduce((acc, val) => {
        return acc.concat(Array.isArray(val)? flatten(val) : val);
    }, []);
}
  • 优点:代码简洁,同样基于递归思想,借助 reduce 方法实现数组遍历和结果累加,处理逻辑清晰。
  • 缺点:和递归方法一样,在处理深度嵌套数组时,由于递归调用过多,会有栈溢出风险,性能也会受影响。

使用 flat 方法

const flattenedArray = multiDimensionalArray.flat(Infinity);
  • 优点:这是 JavaScript 原生提供的方法,由浏览器引擎底层实现,经过高度优化,性能通常较好。使用 Infinity 作为参数能轻松处理任意嵌套深度的数组,代码简洁高效。
  • 缺点:该方法是 ES2019 引入的新特性,对于一些较旧的浏览器版本可能不支持,需要做兼容性处理。

使用 while 循环和 some 方法

function flatten(arr) {
    let isNested = true;
    while (isNested) {
        isNested = false;
        let newArr = [];
        for (let i = 0; i < arr.length; i++) {
            if (Array.isArray(arr[i])) {
                newArr = newArr.concat(arr[i]);
                isNested = true;
            } else {
                newArr.push(arr[i]);
            }
        }
        arr = newArr;
    }
    return arr;
}
  • 优点:避免了递归调用,不会有栈溢出问题,在处理深度嵌套数组时性能相对稳定。
  • 缺点:代码逻辑相对复杂,需要多次遍历数组和创建新数组,在处理大规模数组时,频繁的数组操作会带来一定的性能开销。

效率总结

  • 小规模、浅嵌套数组:递归方法、reduceconcat 方法代码简洁易读,性能差异不大,可任选其一。
  • 大规模、深嵌套数组flat 方法效率最高,因为它是原生方法,经过优化;若考虑兼容性,while 循环和 some 方法是不错的选择,能避免栈溢出问题。

相关文章:

  • LiteratureReading:[2023] GPT-4: Technical Report
  • xLua_003 Lua访问C#
  • 正确理解虚基类
  • Umi-OCR 实践教程:离线、免费、高效的图像文字识别工具
  • 基于STC89C52的DS1621温度检测系统
  • 【python3】:轻松实现点阵数据 转DXF格式
  • ROS2 常用命令详解
  • 第37周:文献阅读
  • java查询es超过10000条数据
  • 内容中台实施效能跃迁方法论
  • Python实现ONNXRuntime推理YOLOv11模型
  • AI 如何重塑数据湖的未来
  • git原理与常用命令及其使用
  • 数学建模:MATLAB卷积神经网络
  • 【嵌入式学习】触发器 - ADC - DAC
  • 微信 MMTLS 协议详解(五):加密实现
  • 【嵌入式硬件测试之道连载之第三章:核心处理器的选型与应用】
  • IS-IS原理与配置
  • Nexus L2 L3基本配置
  • 【Java SE】抽象类/方法、模板设计模式
  • 中美经贸高层会谈在瑞士日内瓦开始举行
  • 2025年4月份CPI环比由降转涨,核心CPI涨幅稳定
  • 巴基斯坦称成功拦截印度导弹,空军所有资产安全
  • 理财经理泄露客户信息案进展:湖南省检受理申诉,证监会交由地方监管局办理
  • 胳膊一抬就疼,炒菜都成问题?警惕这种“炎症”找上门
  • 警惕“全网最低价”等宣传,市监总局和中消协发布直播消费提示