JavaScript中实现对象复制的终极指南

 更新时间:2025年11月09日 08:38:39   作者:Zyx2007  
在 JavaScript 开发中,我们经常需要复制一个对象或数组,当我们处理像用户列表、配置项、表单数据等复杂结构时,浅拷贝(shallow copy)常常无法满足需求,真正的解决方案是深拷贝,本文将从内存原理出发,深入剖析深拷贝的必要性、实现方式及其局限性

引言:一个看似简单却暗藏玄机的问题

在 JavaScript 开发中,我们经常需要复制一个对象或数组。然而,一句简单的 const newData = oldData 往往会带来意想不到的副作用——修改新数据竟然影响了原始数据!这背后的原因,正是 JavaScript 内存模型与引用机制的本质体现。

当我们处理像用户列表、配置项、表单数据等复杂结构时,浅拷贝(shallow copy)常常无法满足需求。真正的解决方案是深拷贝(deep clone) ——创建一个与原对象完全独立、互不影响的新对象。本文将从内存原理出发,深入剖析深拷贝的必要性、实现方式及其局限性,帮助开发者彻底掌握这一核心技能。

一、内存模型:理解“引用”为何危险

要理解深拷贝,必须先理解 JavaScript 的内存分配机制。

1.1 栈内存 vs 堆内存

栈内存(Stack) :存储基本数据类型(如 numberstringboolean)。变量直接保存值,赋值即值拷贝

let a = 1;
let b = a; // b 获得 a 的副本
b = 2;
console.log(a); // 1(不受影响)

堆内存(Heap) :存储引用类型(如 objectarrayfunction)。变量保存的是指向堆内存的地址,赋值即引用拷贝

const users = [{ name: '张三' }];
const data = users; // data 与 users 指向同一块堆内存
data[0].name = '李四';
console.log(users[0].name); // '李四'(原始数据被意外修改!)

这种设计使得复杂数据结构可以动态扩展(如 users.push(...)),但也带来了“共享引用”的风险。

1.2 浅拷贝的局限性

常见的“复制”方法如展开运算符(...)、Object.assign() 都只是浅拷贝

const users = [{ id: 1, name: '张三', hobbies: ['篮球'] }];
const shallowCopy = [...users];
shallowCopy[0].hobbies.push('足球');
console.log(users[0].hobbies); // ['篮球', '足球'] —— 原始数据被污染!

原因在于:浅拷贝只复制了对象的第一层属性,而嵌套的对象/数组仍然共享引用。

二、深拷贝:彻底隔离数据的唯一途径

深拷贝的目标是:递归复制所有层级的属性,确保新旧对象在内存中完全独立

2.1 JSON 方法:最简单的深拷贝

利用 JavaScript 内置的序列化能力,是最常用的深拷贝方案:

const users = [
  { id: 1, name: '张三', hometown: '北京' },
  { id: 2, name: '李四', hometown: '上海' }
];

// 序列化 → 字符串
const jsonString = JSON.stringify(users);
// 反序列化 → 全新对象
const deepCopy = JSON.parse(jsonString);

deepCopy[0].hobbies = ['篮球', '足球'];
console.log(users[0].hobbies);   // undefined(未受影响)
console.log(deepCopy[0].hobbies); // ['篮球', '足球']

优点

  • 代码简洁,一行搞定
  • 自动处理任意深度的嵌套结构
  • 性能较好(底层由 V8 优化)

缺点

  • 无法处理函数、undefinedSymbolDateRegExp 等特殊类型
  • 会忽略对象的原型链(constructor 丢失)
  • 无法处理循环引用(会报错)

因此,JSON 方法适用于纯数据对象(如 API 返回的 JSON 数据),但不适用于包含方法或复杂类型的对象。

2.2 手写递归深拷贝:全面但复杂

为了克服 JSON 方法的局限,我们可以手动实现递归深拷贝:

function deepClone(obj, hash = new WeakMap()) {
  // 处理 null、undefined、基本类型
  if (obj === null || typeof obj !== 'object') return obj;
  
  // 处理 Date
  if (obj instanceof Date) return new Date(obj);
  
  // 处理 RegExp
  if (obj instanceof RegExp) return new RegExp(obj);
  
  // 防止循环引用
  if (hash.has(obj)) return hash.get(obj);
  
  // 创建新实例
  const cloned = new obj.constructor();
  hash.set(obj, cloned);
  
  // 递归拷贝所有属性
  for (let key in obj) {
    if (obj.hasOwnProperty(key)) {
      cloned[key] = deepClone(obj[key], hash);
    }
  }
  
  return cloned;
}

这个版本支持:

  • 函数、日期、正则表达式
  • 循环引用检测(通过 WeakMap
  • 保留原型链

但实现复杂,且性能不如 JSON 方法。

三、实战场景:何时必须使用深拷贝?

3.1 状态管理(React/Vue)

在前端框架中,状态变更必须返回新对象,否则视图不会更新:

//  错误:直接修改原状态
state.users[0].name = '新名字';

//  正确:使用深拷贝创建新状态
const newState = JSON.parse(JSON.stringify(state));
newState.users[0].name = '新名字';
setState(newState);

3.2 表单回滚与撤销功能

当用户编辑表单时,需要保留原始数据用于“取消”操作:

const originalData = JSON.parse(JSON.stringify(formData));
// 用户修改 formData...
// 点击“取消”时
formData = JSON.parse(JSON.stringify(originalData));

四、深拷贝的边界与替代方案

4.1 何时不需要深拷贝?

  • 数据是扁平结构(无嵌套对象)
  • 只读数据(不会被修改)
  • 性能敏感场景(深拷贝开销大)

此时,浅拷贝或直接引用更高效。

4.2 结构化克隆(Structured Clone)

现代浏览器支持 structuredClone() API(ES2022):

const deepCopy = structuredClone(users);

它支持更多类型(包括 DateRegExpMapSet),但仍不支持函数和循环引用。

五、最佳实践建议

  1. 优先使用 JSON 方法:适用于 90% 的纯数据场景
  2. 明确数据边界:只对可能被修改的复杂对象进行深拷贝
  3. 避免过度拷贝:深拷贝性能开销大,不要滥用
  4. 测试边界情况:确保深拷贝方案能处理你的实际数据结构

结语:深拷贝不仅是技术,更是思维

深拷贝问题的本质,是对数据所有权副作用控制的理解。在函数式编程日益流行的今天,“不可变性”已成为构建可靠系统的基石。

掌握深拷贝,不仅是为了写出正确的代码,更是为了培养一种防御性编程思维:永远假设数据会被修改,永远确保自己的操作不会影响他人。

“在 JavaScript 的世界里,共享引用是默认,独立拷贝是选择。”
—— 而深拷贝,正是我们做出正确选择的有力工具。

以上就是JavaScript中实现对象复制的终极指南的详细内容,更多关于JavaScript对象复制的资料请关注脚本之家其它相关文章!

相关文章

  • javascript实现扫雷简易版

    javascript实现扫雷简易版

    这篇文章主要为大家详细介绍了javascript实现扫雷简易版,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-08-08
  • js项目中前端如何实现无感刷新token

    js项目中前端如何实现无感刷新token

    本文主要介绍了js项目中前端如何实现无感刷新token,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-07-07
  • js实现新浪微博首页效果

    js实现新浪微博首页效果

    现在微博都有一个不错的效果就是会实时的动态滚动并显示最新的信息了,下面我来给大家介绍仿新浪微博大厅首页滚动效果,希望此方法对大家会有帮助。
    2015-10-10
  • 详解使用JWT实现单点登录(完全跨域方案)

    详解使用JWT实现单点登录(完全跨域方案)

    这篇文章主要介绍了详解使用JWT实现单点登录(完全跨域方案),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • js读取cookie方法总结

    js读取cookie方法总结

    这篇文章主要介绍了js读取cookie方法,实例总结了四种常用的读取方法,包含了对cookie数据的分割与遍历操作,是非常实用的技巧,需要的朋友可以参考下
    2014-10-10
  • JS+CSS实现鼠标滑过时动态翻滚的导航条效果

    JS+CSS实现鼠标滑过时动态翻滚的导航条效果

    这篇文章主要介绍了JS+CSS实现鼠标滑过时动态翻滚的导航条效果,涉及JavaScript动态设置css样式动画过度效果的技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-09-09
  • JavaScript数组去重由慢到快由繁到简(优化篇)

    JavaScript数组去重由慢到快由繁到简(优化篇)

    本文给大家介绍通过indexof去重,hash去重,排序后去重及set去重由慢到快有繁到简的方法给大家介绍了js数组去重的方法,非常不错,具有参考借鉴价值,感兴趣的朋友一起看看吧
    2016-08-08
  • 小试JavaScript多线程

    小试JavaScript多线程

    这两天一直在弄ajax,用多了才发现了ajax 的cache问题,请求了好多次,得到了确是相同的结果,经常我想在请求的同时去做一些其它的事情,我在想javascript里有没有办法用多线程,经过在网上查找找到了结果。
    2008-11-11
  • 执行iframe中的javascript方法

    执行iframe中的javascript方法

    如何调用执行iframe中的方法?如下
    2008-10-10
  • JavaScript仿微博输入框效果(案例分析)

    JavaScript仿微博输入框效果(案例分析)

    这篇文章给大家分享一个小的JavaScript的案例,就是模仿微博输入框的效果,非常不错,对微博输入框效果感兴趣的朋友通过本文学习吧
    2016-12-12

最新评论