news 2026/6/9 23:52:22

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/6/10 14:56:00

SSM288的美好生活分享九宫格日志网

目录平台定位与特色核心功能用户群体与使用场景设计理念与交互数据与隐私开发技术源码文档获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;关于SSM288的美好生活分享九宫格日志网&#xff0c;以下为综合整理后的摘要内容&#xff1a; 平台定位与…

作者头像 李华
网站建设 2026/5/30 20:19:01

什么是仓库管理系统 WMS?它到底有什么用?

谢邀。什么是仓库管理系统WMS&#xff1f;它到底有什么用&#xff1f;这个问题&#xff0c;其实在不少企业里都被反复问过。但有意思的是—— 你真的去问一位仓管&#xff0c;答案往往是&#xff1a;“你们仓库有没有WMS&#xff1f;”“有啊&#xff0c;有 Excel。”这其实是很…

作者头像 李华
网站建设 2026/6/10 12:27:20

聊聊超靠谱的UDS协议栈

UDS协议栈 满足基本的Autosar接口标准&#xff0c;有说明文档&#xff0c;代码通俗易懂&#xff0c;经过量产验证&#xff0c;质量绝对有保障&#xff01;&#xff01;代码为平台代码&#xff0c;移植简单&#xff01;最近在研究汽车电子相关的技术&#xff0c;发现了一款超赞的…

作者头像 李华
网站建设 2026/6/10 12:28:05

S7-1200 控制 5 轴伺服程序案例分享

S7-1200控制5轴伺服程序案例。1.PTO伺服轴脉冲定位控制功能应用速度模式应用扭矩模式应用。 2.程序为结构化编程,每一功能为模块化设计,具有一个项目都有的功能:自动_手动_单步_暂停后原位置继续运行_轴断电保持_报警功能_气缸运行及报警. 3.每个功能块可以无数次重复调用&…

作者头像 李华
网站建设 2026/6/10 12:26:23

医疗小程序音视频问诊门诊医院药房系统开发漫谈

医疗小程序音视频问诊门诊医院药房系统开发源码在当今数字化医疗的浪潮下&#xff0c;医疗小程序音视频问诊结合门诊、医院药房功能的系统&#xff0c;无疑是为患者和医疗机构搭建了一座高效便捷的桥梁。今天咱就来聊聊这系统开发源码背后的事儿。 音视频问诊模块 音视频问诊功…

作者头像 李华
网站建设 2026/6/10 12:28:14

C#上位机与台达DVP系列Modbus 485通信实战

C#上位机&#xff0c;台达DVP系列modbus485通信例子。 例子简单易看懂。 自己写的程序。在自动化控制领域&#xff0c;上位机与下位机的通信至关重要。今天就来分享一个用C#编写的上位机与台达DVP系列通过Modbus 485进行通信的例子&#xff0c;希望能帮助到正在研究相关内容的小…

作者头像 李华