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
相关推荐
戴西软件9 小时前
戴西软件入选2026年安徽省制造业数智化转型服务商名单
java·大数据·服务器·前端·人工智能
薛定猫AI10 小时前
【深度解析】从 Antigravity 更新看 Agent IDE 的工程化演进:权限、沙盒、MCP 与模型治理
前端·javascript·ide
漂流瓶jz17 小时前
总结CSS组件化演进之路:命名规范/CSS Modules/CSS in JS/原子化CSS
前端·javascript·css
踩着两条虫17 小时前
「AI + 低代码」的可视化设计器
开发语言·前端·低代码·设计模式·架构
Jagger_18 小时前
项目上线忙碌结束之后,为什么总想找点事做?
前端
GalenZhang88818 小时前
OpenClaw 配置多个飞书账号实战指南
前端·chrome·飞书·openclaw
steven~~~18 小时前
为什么mq报错
javascript
萌新小码农‍19 小时前
python装饰器
开发语言·前端·python
threelab20 小时前
Three.js 初中数学函数可视化 | 三维可视化 / AI 提示词
开发语言·前端·javascript·人工智能·3d·着色器
爱学习的程序媛20 小时前
浏览器工作原理全景解析
前端·浏览器·web