当前位置: 首页 > news >正文

10. 怎么实现深拷贝?

总结

实现深拷贝的方式有多种,选择应根据具体需求:

  • 简单场景:使用 JSON.parse(JSON.stringify(obj))
  • 学习/教学:使用递归实现 + 循环引用处理
  • 生产环境:优先使用 lodash.cloneDeep
  • 浏览器端(现代):使用 structuredClone

建议:在开发中优先使用成熟的库(如 lodash),避免重复造轮子。如需自定义实现,应考虑特殊类型、循环引用、性能等问题。


概述

在 JavaScript 中,深拷贝是指创建一个新对象,使其与原对象完全独立,互不影响。与之相对的是浅拷贝,只复制引用地址,原对象和新对象共享内部引用的数据。

深拷贝常用于:

  • 状态快照保存(如撤销/重做功能)
  • 数据隔离(避免修改原始数据)
  • 跨组件通信时的数据传递

一、常见的深拷贝方式

1. 使用 JSON.parse(JSON.stringify(obj))(简单但有局限)

原理:

将对象序列化为 JSON 字符串,再解析为新对象。

示例:
const obj = { a: 1, b: { c: 2 } };
const copy = JSON.parse(JSON.stringify(obj));copy.b.c = 3;
console.log(obj.b.c); // 2(原对象未被修改)
优点:
  • 简洁、无需额外代码
  • 支持大多数基础结构
缺点:
限制说明
不支持函数、undefined会被忽略
不支持循环引用会报错
不支持 Symbol 属性键会被忽略
日期对象会被转为字符串无法还原为 Date 对象
正则表达式等特殊对象也会丢失RegExp

2. 递归实现深拷贝(基础实现)

示例:
function deepClone(obj) {if (obj === null || typeof obj !== "object") return obj;const copy = Array.isArray(obj) ? [] : {};for (const key in obj) {if (obj.hasOwnProperty(key)) {copy[key] = deepClone(obj[key]);}}return copy;
}
优点:
  • 可以处理对象、数组
  • 可扩展性强
缺点:
  • 不处理循环引用会栈溢出
  • 无法复制函数、DateRegExp 等特殊对象

3. 使用第三方库(推荐)

(1) lodashcloneDeep
npm install lodash
import _ from "lodash";const obj = { a: 1, b: { c: 2 } };
const copy = _.cloneDeep(obj);
(2) structuredClone(浏览器原生 API,现代浏览器支持)
const obj = { a: 1, b: { c: 2 } };
const copy = structuredClone(obj);

✅ 支持:DateMapSetArrayBufferError
❌ 不支持:函数、undefined、某些循环引用


二、处理循环引用的深拷贝实现

function deepClone(obj, visited = new Map()) {if (obj === null || typeof obj !== "object") return obj;if (visited.has(obj)) {return visited.get(obj); // 防止循环引用}const copy = Array.isArray(obj) ? [] : {};visited.set(obj, copy);for (const key in obj) {if (obj.hasOwnProperty(key)) {copy[key] = deepClone(obj[key], visited);}}return copy;
}

三、不同深拷贝方式对比

方法支持类型循环引用函数支持日期支持正则支持性能推荐场景
JSON.parse✅ 对象/数组⭐⭐⭐⭐简单对象,无特殊类型
递归实现✅ 对象/数组❌(需手动处理)⭐⭐学习用途
lodash.cloneDeep✅ 多种类型⭐⭐⭐⭐⭐生产环境
structuredClone✅ 多种类型⭐⭐⭐⭐浏览器端
手动封装(带类型判断)✅ 自定义✅(可实现)✅(可实现)✅(可实现)✅(可实现)⭐⭐⭐定制化需求

四、深拷贝的注意事项

说明
循环引用必须使用 MapWeakMap 缓存已拷贝对象
特殊对象DateRegExpMapSet 需要单独处理
函数通常不需要拷贝,直接返回原引用即可
Symbol 类型键需要用 Reflect.ownKeys 获取
原型链上的属性通常不需要拷贝,除非特别要求
性能优化深拷贝可能影响性能,应避免频繁调用

http://www.lryc.cn/news/614845.html

相关文章:

  • ABP VNext + Apache Kafka Exactly-Once 语义:金融级消息一致性实战
  • VSCode添加Python、Java注释技巧、模板
  • 笔试——Day33
  • java web项目入门了解
  • 微信原生小程序 Timeline 组件实现
  • 在Word和WPS文字中快速拆分、合并表格
  • JavaWeb03——javascript基础语法
  • C++-AVL树
  • 微软将于 10 月停止混合 Exchange 中的共享 EWS 访问
  • SOLi-LABS Page-3 (Stacked injections) --39-53关
  • 使用 Vuepress + GitHub Pages 搭建项目文档(2)- 使用 GitHub Actions 工作流自动部署
  • 如何解决 Vue 项目启动时出现的 “No such module: http_parser” 错误问题
  • 2G内存的服务器用宝塔安装php的fileinfo拓展时总是卡死无法安装成功的解决办法
  • 企业级web应用服务器TOMCAT入门详解
  • kettle插件-kettle MinIO插件,轻松解决文件上传到MinIO服务器
  • 解决本地连接服务器ollama的错误
  • 大语言模型提示工程与应用:大语言模型对抗性提示安全防御指南
  • LLVM编译器入门
  • Java基础-TCP通信单服务器接受多客户端
  • 关于开发语言的一些效率 从堆栈角度理解一部分c java go python
  • 软考 系统架构设计师系列知识点之杂项集萃(119)
  • 数据结构(9)——排序
  • QT第三讲- 机制、宏、类库模块
  • 数字图像处理基础——opencv库(Python)
  • 算法_python_牛客华为机试笔记_01
  • 【Python 高频 API 速学 ③】
  • RecyclerView 中 ViewHolder
  • TDengine IDMP 快速体验(1. 通过云服务)
  • 【CVPR2025】计算机视觉|PX:让模型训练“事半功倍”!
  • vscode/trae 的 settings.json 中配置 latex 的一些记录