news 2026/4/25 17:25:47

JavaScript 数组合并性能优化:扩展运算符 vs concat vs 循环 push

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
JavaScript 数组合并性能优化:扩展运算符 vs concat vs 循环 push

在日常开发中,我们经常需要合并数组,比如批量导入数据、分页加载列表、处理大量日志等场景。当数组规模较小时,用什么方法都差不多;但当数组达到成千上万条时,选择不当的方法可能会导致栈溢出内存飙升

今天我们就来聊聊 JavaScript 中三种常见的数组合并方式,看看它们各自的优缺点,以及在不同场景下应该如何选择。

三种数组合并方式

1. 扩展运算符 + push:arr.push(...otherArr)

这是最"爽"的写法,一行代码搞定:

constarr1=[1,2,3]constarr2=[4,5,6]arr1.push(...arr2)console.log(arr1)// [1, 2, 3, 4, 5, 6]

优点:

  • 代码简洁,可读性强
  • 语法现代,符合 ES6+ 风格
  • 适合小数组快速合并

缺点:

  • 当数组元素超过约 65536 个时,会触发 JavaScript 引擎的最大参数数量限制,导致栈溢出错误
  • 扩展运算符需要先展开数组,会消耗额外内存
  • 大数组性能较差

适用场景:

  • 小数组(< 1000 个元素)
  • 快速脚本开发
  • 数量可控的场景

2. concat 方法:arr.concat(otherArr)

这是最"安全"的写法:

constarr1=[1,2,3]constarr2=[4,5,6]constmerged=arr1.concat(arr2)console.log(merged)// [1, 2, 3, 4, 5, 6]

优点:

  • 不会导致栈溢出,适合大数组
  • 语义清晰,代码可读性好
  • 返回新数组,不修改原数组(函数式编程友好)
  • 性能稳定,适合生产环境

缺点:

  • 会创建新数组,占用额外内存
  • 对于超大数组,内存占用会翻倍

适用场景:

  • 中大数组(> 1000 个元素)
  • 需要保持原数组不变
  • 函数式编程风格
  • 生产环境安全要求高

3. 循环 push:for (const item of arr) { target.push(item) }

这是最"稳"的写法:

constarr1=[1,2,3]constarr2=[4,5,6]for(constitemofarr2){arr1.push(item)}console.log(arr1)// [1, 2, 3, 4, 5, 6]

优点:

  • 最安全,绝对不会栈溢出
  • 内存占用最低(原地修改,不创建新数组)
  • 性能稳定,适合超大数组
  • 兼容性最好

缺点:

  • 代码相对冗长
  • 需要手动写循环
  • 会修改原数组

适用场景:

  • 超大数组(> 10000 个元素)
  • 内存敏感场景
  • 长时间运行的服务端任务
  • 需要最高性能的场景

性能对比表

方法栈溢出风险内存占用性能代码简洁度适用数组大小
push(...arr)⚠️ 高(>65536)差(大数组)⭐⭐⭐⭐⭐< 1000
concat()✅ 无中高⭐⭐⭐⭐> 1000
循环push()✅ 无最好⭐⭐⭐> 10000

实际案例对比

让我们看看在不同规模下的表现:

小数组(100 个元素)

// 三种方法都可以,性能差异可忽略constsmall=Array.from({length:100},(_,i)=>i)// 方法1:扩展运算符arr1.push(...small)// ✅ 推荐// 方法2:concatconstmerged=arr1.concat(small)// ✅ 也可以// 方法3:循环for(constitemofsmall)arr1.push(item)// ✅ 也可以,但没必要

中数组(10000 个元素)

constmedium=Array.from({length:10000},(_,i)=>i)// 方法1:扩展运算符arr1.push(...medium)// ⚠️ 可能栈溢出!// 方法2:concatconstmerged=arr1.concat(medium)// ✅ 推荐// 方法3:循环for(constitemofmedium)arr1.push(item)// ✅ 也可以

大数组(100000 个元素)

constlarge=Array.from({length:100000},(_,i)=>i)// 方法1:扩展运算符arr1.push(...large)// ❌ 几乎肯定会栈溢出!// 方法2:concatconstmerged=arr1.concat(large)// ⚠️ 可以,但内存占用高// 方法3:循环for(constitemoflarge)arr1.push(item)// ✅ 强烈推荐

进阶优化技巧

1. 分批处理超大数组

当数组特别大时,可以分批处理,避免一次性操作:

functionbatchConcat(target,source,batchSize=5000){for(leti=0;i<source.length;i+=batchSize){constbatch=source.slice(i,i+batchSize)target.push(...batch)// 小批次可以用扩展运算符}returntarget}// 使用consthuge=Array.from({length:100000},(_,i)=>i)constresult=[]batchConcat(result,huge)// 安全处理超大数组

2. 使用 Array.from 和展开运算符组合

对于需要转换的场景:

// 不推荐:可能栈溢出constmapped=arr.map(x=>x*2)result.push(...mapped)// 推荐:使用 concatconstmapped=arr.map(x=>x*2)result=result.concat(mapped)// 或者:直接循环for(constitemofarr){result.push(item*2)}

3. 流式处理(迭代器)

如果数据来自流或生成器,可以边读边处理:

asyncfunction*fetchData(){// 模拟分页数据流for(leti=0;i<10;i++){yieldawaitfetchPage(i)}}constresult=[]forawait(constpageoffetchData()){// 流式处理,内存占用低result.push(...page)// 每页数据量小,可以用扩展运算符}

常见错误示例

❌ 错误示例 1:批量导入时使用扩展运算符

// 批量导入 Excel,可能有上万行数据constexcelData=[]sheets.forEach((sheet)=>{constcurrent=xlsx.utils.sheet_to_json(workbook.Sheets[sheet])excelData.push(...current)// ❌ 如果数据量大,会栈溢出!})

正确做法:

constexcelData=[]sheets.forEach((sheet)=>{constcurrent=xlsx.utils.sheet_to_json(workbook.Sheets[sheet])// ✅ 使用 concat 或循环excelData=excelData.concat(current)// 或者// for (const row of current) excelData.push(row)})

❌ 错误示例 2:分页加载时使用扩展运算符

// 分页加载,累计可能有大量数据constloadMore=async()=>{constnewData=awaitfetchData(offset)allData.push(...newData)// ❌ 随着数据累积,可能栈溢出}

正确做法:

constloadMore=async()=>{constnewData=awaitfetchData(offset)// ✅ 使用 concatallData=allData.concat(newData)// 或者// for (const item of newData) allData.push(item)}

最佳实践建议

  1. 小数组(< 1000):随意,push(...arr)最简洁
  2. 中数组(1000 - 10000):优先使用concat(),安全可靠
  3. 大数组(> 10000):使用循环push(),性能最优
  4. 不确定大小时:保守选择concat()或循环push()
  5. 需要保持原数组不变:使用concat()
  6. 内存敏感场景:使用循环push(),原地修改

总结

数组合并看似简单,但在处理大规模数据时,选择合适的方法至关重要:

  • 扩展运算符push(...arr):适合小数组,代码简洁,但大数组会栈溢出
  • concat():适合中大数组,安全可靠,但会创建新数组
  • 循环push():适合超大数组,性能最优,内存占用最低

记住一个原则:不确定数组大小时,选择更安全的方法。在生产环境中,concat()和循环push()是更稳妥的选择。

希望这篇文章能帮助你在实际开发中避免栈溢出的坑!如果觉得有用,欢迎点赞收藏~

参考

  • MDN: Array.prototype.push()
  • MDN: Array.prototype.concat()
  • JavaScript 引擎参数限制
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/25 10:40:42

python项目打包为镜像

1.生成 requirements.txt 在项目根目录下,使用 pipreqs生成依赖文件,确保镜像构建时安装正确的包 pip install pipreqs pipreqs . --encoding=utf8 --force 2.编写 Dockerfile # 使用官方 Python 轻量级镜像 FROM python:3.11-slim# 设置容器内工作目录 WORKDIR /app# 复制…

作者头像 李华
网站建设 2026/4/22 1:46:06

Spark Streaming与大数据批处理的结合应用

Spark Streaming与大数据批处理的结合应用:实时与离线的完美搭档 关键词:Spark Streaming、大数据批处理、流批结合、实时计算、离线分析 摘要:在大数据时代,企业既需要实时掌握业务动态(比如用户刚下单的商品),也需要挖掘历史规律(比如过去一年的销售趋势)。Spark St…

作者头像 李华
网站建设 2026/4/25 17:25:15

优化提示内容交互设计的9个实用技巧

优化提示内容交互设计的9个实用技巧&#xff1a;让AI更懂你的“说话之道” 一、引入与连接&#xff1a;为什么你需要学“提示设计”&#xff1f; 清晨&#xff0c;你打开ChatGPT&#xff0c;输入&#xff1a;“帮我写篇关于秋天的文章。”半小时后&#xff0c;你看着屏幕上那篇…

作者头像 李华
网站建设 2026/4/25 7:47:50

欧姆龙CP1H + CIF11与欧姆龙E5cc温控器通讯程序分享

欧姆龙CP1HCIF11与欧姆龙E5cc温控器通讯程序 功能&#xff1a;全新原创可直接应用生产程序。 通过昆仑通态触摸屏&#xff0c;串口网关模式&#xff0c;欧姆龙CP1H的CIF11通讯板&#xff0c;实现对欧姆龙E5CC温控器 设定温度值&#xff0c;读取实际温度&#xff0c;设定探头类型…

作者头像 李华
网站建设 2026/4/25 17:24:44

Kiro教程(三)| Kiro 实战与最佳实践

Kiro教程&#xff08;三&#xff09;| Kiro 实战与最佳实践 Kiro 实战与最佳实践案例 1&#xff1a;从零开发 Todo 应用&#xff08;Spec 模式&#xff09;项目要求Step 1&#xff1a;创建项目并配置 SteeringStep 2&#xff1a;启动 SpecStep 3&#xff1a;审核三阶段文档 案例…

作者头像 李华