JavaScript 深浅拷贝详解

JavaScript 深浅拷贝详解

一、为什么需要拷贝?

JavaScript 中数据类型分为两类:

  • 基本类型NumberStringBooleannullundefinedSymbolBigInt):存储在栈中,赋值即复制值。
  • 引用类型ObjectArrayFunction 等):栈中存的是地址,堆中存的是数据,赋值复制的是地址。
js 复制代码
const a = { name: 'Tom' };
const b = a;
b.name = 'Jerry';
console.log(a.name); // 'Jerry'  ← a 也被改了

b = a 仅复制了引用地址,两者指向同一对象。要避免这种联动,就需要"拷贝"。


二、浅拷贝(Shallow Copy)

只拷贝第一层属性。如果属性值是引用类型,仍然共享地址。

1. Object.assign

js 复制代码
const obj = { a: 1, b: { c: 2 } };
const copy = Object.assign({}, obj);

copy.a = 99;        // 不影响 obj
copy.b.c = 99;      // 影响 obj.b.c ← 第二层共享

2. 扩展运算符 ...

js 复制代码
const copy = { ...obj };
const arrCopy = [...arr];

行为与 Object.assign 一致,只拷贝一层。

3. 数组方法 slice / concat

js 复制代码
const arr = [1, [2, 3]];
const copy = arr.slice();
copy[1][0] = 99; // arr 也变 [1, [99, 3]]

手写一个浅拷贝

js 复制代码
function shallowCopy(target) {
  if (target === null || typeof target !== 'object') return target;
  const result = Array.isArray(target) ? [] : {};
  for (const key in target) {
    if (Object.prototype.hasOwnProperty.call(target, key)) {
      result[key] = target[key];
    }
  }
  return result;
}

三、深拷贝(Deep Copy)

递归拷贝所有层级,新旧对象完全独立。

1. JSON.parse(JSON.stringify(obj))

最常见的"偷懒"写法:

js 复制代码
const copy = JSON.parse(JSON.stringify(obj));

优点: 简洁。 缺点(坑很多):

  • undefinedfunctionSymbol 会被丢弃或转 null
  • Date 变成字符串
  • RegExpMapSet 变成 {}
  • NaNInfinity 变成 null
  • 不能处理循环引用(直接报错)

2. structuredClone(推荐 ✅)

浏览器和 Node.js 17+ 原生支持:

js 复制代码
const copy = structuredClone(obj);

支持 DateMapSetRegExp、循环引用等,目前最优解。 不能拷贝函数和 DOM 节点。

3. 第三方库

  • lodash.cloneDeep(obj) --- 兼容性好,处理边界场景全面。

4. 手写一个深拷贝

要点:递归 + 处理引用类型 + 防循环引用。

js 复制代码
function deepClone(target, hash = new WeakMap()) {
  if (target === null || typeof target !== 'object') return target;
  if (target instanceof Date) return new Date(target);
  if (target instanceof RegExp) return new RegExp(target);

  if (hash.has(target)) return hash.get(target); // 循环引用

  const result = Array.isArray(target) ? [] : {};
  hash.set(target, result);

  Reflect.ownKeys(target).forEach(key => {
    result[key] = deepClone(target[key], hash);
  });

  return result;
}

关键点:

  • WeakMap 缓存已拷贝对象,解决循环引用
  • Reflect.ownKeys 同时拿到字符串键和 Symbol
  • 单独处理 DateRegExp 等特殊对象

四、对比总结

方式 深/浅 函数 Date 循环引用 Symbol
Object.assign / ...
JSON 大法 ❌字符串 ❌报错
structuredClone
lodash.cloneDeep
手写 deepClone 视实现 视实现

五、选型建议

  • 只需一层独立 → 扩展运算符 ...
  • 现代环境深拷贝structuredClone
  • 数据是纯 JSONJSON.parse(JSON.stringify(...)) 够用
  • 复杂场景 / 老环境lodash.cloneDeep
  • 理解原理 / 面试 → 手写 deepClone
相关推荐
云水一下43 分钟前
TypeScript 从零基础到精通(五):高级类型与泛型
前端·javascript·typescript
counterxing1 小时前
vibe coding 之后,我更不想打字了
前端·agent·ai编程
云水一下1 小时前
TypeScript 从零基础到精通(六):类型声明与模块化
javascript·typescript
copyer_xyf1 小时前
Python 模块与包的导入导出
前端·后端·python
研☆香1 小时前
es6新特性功能介绍(四)
前端·ecmascript·es6
微扬嘴角2 小时前
React篇1--JSX语法规则、组件、组件实例的3大特性
前端·react.js·前端框架
copyer_xyf2 小时前
Python venv 虚拟环境
前端·后端·python
无聊的老谢2 小时前
Vue 3 + TypeScript 构建大型电信运维平台的前端架构设计
前端·vue.js·typescript
xiaofeichaichai2 小时前
Map / Set / WeakMap / WeakSet
前端·javascript
李可以量化2 小时前
成交量的终极量化策略:价量共振指标完整实现(下篇)
前端·数据库·人工智能