展开运算符...用于将可迭代对象摊开为独立元素,不直接操作普通对象;数组浅拷贝、合并时需注意引用共享;对象展开仅处理自有可枚举属性,同名属性后覆盖前,不递归、不处理不可枚举项及原型链。

展开运算符 ... 是什么?它不是万能的复制工具
JavaScript 的 ...(展开运算符)本质是「将可迭代对象(如数组、字符串、Map、Set)或类数组对象「摊开」成独立元素」。它不直接操作对象,对普通对象使用时会报错 TypeError: Found non-callable @@iterator,除非该对象实现了 Symbol.iterator。注意:它和剩余参数(...args)语法相同但语义相反——一个「往外摊」,一个「往里收」。
数组合并与浅拷贝:别用 push(...arr) 往原数组塞
常见误用是想把一个数组所有元素推入另一个数组,却写成 target.push(...source)。这确实可行,但要注意:push 会修改原数组;若只想生成新数组,更推荐 [...target, ...source]。另外,[...arr] 只做一层浅拷贝,嵌套对象/数组仍共享引用。
const a = [1, 2]; const b = [...a, 3, 4]; // [1, 2, 3, 4]const c = [[1], [2]]; const d = [...c]; d[0].push(9); console.log(c); // [[1, 9], [2]] —— 原数组被改了- 替代深拷贝?不行。要用
JSON.parse(JSON.stringify(arr))或结构化克隆(structuredClone)
函数调用传参:比 apply 更直观,但不能替代 bind
展开运算符让动态传参变得自然,尤其适合已知参数在数组中时。但它无法像 bind 那样预设部分参数并返回新函数。
Math.max(...[3, 7, 2]) // 7,等价于 Math.max(3, 7, 2)const args = ['a', 'b']; func(...args); // ✅const bound = func.bind(null, ...args); // ❌ 语法错误 —— bind 不接受展开表达式作为参数列表- 正确做法:
const bound = (...rest) => func(...args, ...rest);
对象展开:只处理自有可枚举属性,且顺序可能影响结果
对象展开({...obj})实际是按属性遍历顺序(ES2015+ 规定:数字键升序 → 字符串键插入顺序 → Symbol 键插入顺序)逐个赋值。后出现的同名属性会覆盖前面的。
立即学习“Java免费学习笔记(深入)”;
const a = { x: 1, y: 2 }; const b = { y: 3, z: 4 }; const c = { ...a, ...b }; // { x: 1, y: 3, z: 4 }- 不可枚举属性(如
Object.defineProperty(obj, 'hidden', { value: 5, enumerable: false }))不会被展开 - 原型链上的属性也不会被展开 —— 它只取对象自身的可枚举属性
- 遇到 getter/setter?会执行 getter 获取值,但 setter 不会被触发











