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
相关推荐
六bring个六2 小时前
opencv简单操作(一)
前端·webpack·node.js
小陈同学呦2 小时前
fetch和axios区别
前端·javascript
森叶2 小时前
Electron 实战:用 utilityProcess 开子进程,去端口化承载协议处理,并由主进程拦截渲染请求后统一中转
前端·javascript·electron
精益数智工坊2 小时前
红牌作战是什么?红牌作战的实施步骤与核心要点
大数据·运维·前端·人工智能·精益工程
techdashen2 小时前
Cloudflare HTML 解析器的十年演化史(一)
前端·html
ZC跨境爬虫2 小时前
移动端爬虫工具Fiddler完整配置流程:PC+安卓模拟器全覆盖,零基础一次配置成功
android·前端·爬虫·测试工具·fiddler
GISer_Jing2 小时前
前端视角:B端传统配置化现状与AI冲击趋势
前端·人工智能·ai编程
课灵_klhubs3 小时前
课灵h5p-3D 模型 (3D Model)教程
前端·3d·课程设计·教程·课灵·h5p
倾颜3 小时前
接入 MCP 之后,我如何让 Skill 稳定消费 Tool / Resource / Prompt
前端·next.js·mcp