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

ES6 深克隆与浅克隆详解:原理、实现与应用场景

ES6 深克隆与浅克隆详解:原理、实现与应用场景

一、克隆的本质与必要性

在 JavaScript 中,数据分为两大类型:

  • 基本类型:Number、String、Boolean、null、undefined、Symbol、BigInt
  • 引用类型:Object、Array、Function、Date、RegExp 等

克隆的必要性

const original = { a: 1, b: { c: 2 } };
const copy = original; // 引用复制original.a = 10;
console.log(copy.a); // 10 - 修改影响副本original.b.c = 20;
console.log(copy.b.c); // 20 - 深层修改也影响副本

二、浅克隆(Shallow Clone)

1. 概念与特点

  • 只复制对象的第一层属性
  • 嵌套对象仍然是引用关系
  • 适用于简单对象结构

2. ES6 实现方式

2.1 扩展运算符(…)
const obj = { a: 1, b: { c: 2 } };
const shallowClone = { ...obj };obj.b.c = 3;
console.log(shallowClone.b.c); // 3(受影响)
2.2 Object.assign()
const arr = [1, 2, { d: 4 }];
const arrClone = Object.assign([], arr);arr[2].d = 5;
console.log(arrClone[2].d); // 5(受影响)
2.3 Array.slice()
const original = [1, { name: 'Alice' }];
const clone = original.slice();original[1].name = 'Bob';
console.log(clone[1].name); // 'Bob'(受影响)

3. 使用场景

  • 组件 props 传递配置对象
  • 状态管理中的状态快照
  • 简单数据结构的临时拷贝

三、深克隆(Deep Clone)

1. 概念与特点

  • 递归复制对象的所有层级
  • 创建完全独立的内存副本
  • 修改原对象不影响克隆对象

2. 手动实现深克隆

基础递归实现:
function deepClone(source) {// 基本类型直接返回if (source === null || typeof source !== 'object') {return source;}// 处理数组if (Array.isArray(source)) {return source.map(item => deepClone(item));}// 处理对象const clone = {};for (const key in source) {if (source.hasOwnProperty(key)) {clone[key] = deepClone(source[key]);}}return clone;
}// 测试
const obj = { a: 1, b: { c: 2 } };
const deepCopy = deepClone(obj);obj.b.c = 3;
console.log(deepCopy.b.c); // 2(不受影响)
增强版(处理特殊对象):
function enhancedDeepClone(source, map = new WeakMap()) {// 基本类型直接返回if (source === null || typeof source !== 'object') {return source;}// 处理循环引用if (map.has(source)) return map.get(source);// 特殊对象处理switch (source.constructor) {case Date:return new Date(source);case RegExp:return new RegExp(source);case Set:return new Set([...source].map(item => enhancedDeepClone(item, map)));case Map:return new Map([...source].map(([k, v]) => [enhancedDeepClone(k, map), enhancedDeepClone(v, map)]));case ArrayBuffer:return source.slice(0);}// 普通对象和数组const clone = Array.isArray(source) ? [] : {};map.set(source, clone);// 处理Symbol键const symbolKeys = Object.getOwnPropertySymbols(source);const allKeys = [...Object.keys(source), ...symbolKeys];for (const key of allKeys) {clone[key] = enhancedDeepClone(source[key], map);}return clone;
}

3. JSON 序列化法(有局限)

const obj = {date: new Date(),regex: /pattern/g,func: () => console.log('test')
};const jsonClone = JSON.parse(JSON.stringify(obj));console.log(jsonClone);
// {
//   date: "2023-08-15T12:00:00.000Z", // Date转为字符串
//   regex: {},                        // RegExp变为空对象
//   func: null                        // 函数丢失
// }

JSON方法的缺陷

  1. 丢失函数属性
  2. 忽略Symbol键
  3. Date对象转为字符串
  4. RegExp变为空对象
  5. 无法处理循环引用
  6. 忽略undefined值

4. 现代浏览器原生API:structuredClone()

const obj = {date: new Date(),regex: /pattern/g,array: [1, 2, 3],set: new Set([4, 5, 6])
};const clone = structuredClone(obj);console.log(clone);
// {
//   date: Date, // 保持Date对象
//   regex: /pattern/g, // 保持正则
//   array: [1,2,3],
//   set: Set(3) {4,5,6}
// }

支持的数据类型

  • 原始值
  • Array/ArrayBuffer
  • Map/Set
  • Date/RegExp
  • Blob/File/FileList
  • ImageData/ImageBitmap
  • 错误类型(Error, EvalError等)
  • 对象字面量(仅包含以上类型)

不支持的类型

  • Function
  • DOM节点
  • 对象方法
  • 属性描述符/getter/setter
  • 原型链

四、深克隆性能优化

1. 循环引用处理

function deepClone(source, map = new WeakMap()) {// ... 其他代码if (map.has(source)) {return map.get(source);}const target = new constructor();map.set(source, target);// ... 克隆逻辑
}

2. 非递归实现(栈代替递归)

function deepCloneStack(source) {const stack = [];const map = new WeakMap();const target = new source.constructor();stack.push([source, target]);map.set(source, target);while (stack.length) {const [current, parent] = stack.pop();for (const key in current) {if (current.hasOwnProperty(key)) {const value = current[key];// 基本类型直接赋值if (value === null || typeof value !== 'object') {parent[key] = value;continue;}// 处理循环引用if (map.has(value)) {parent[key] = map.get(value);continue;}// 创建新对象const child = new value.constructor();parent[key] = child;map.set(value, child);stack.push([value, child]);}}}return target;
}

五、应用场景与最佳实践

1. 浅克隆适用场景

  • 组件 props 传递配置对象
  • Redux reducer 中的状态更新
  • 函数参数中的对象扩展
  • 无嵌套结构的简单数据对象
// Vue 组件 props 解构
export default {props: ['config'],setup(props) {const localConfig = { ...props.config }; // 浅克隆}
}

2. 深克隆必要场景

  • 状态管理中的初始状态快照
  • 需要完全隔离的缓存数据
  • 撤销/重做功能实现
  • 复杂配置对象的版本管理
  • Web Worker 数据传输
// 在 Web Worker 中处理数据
worker.postMessage(structuredClone(largeDataSet));

3. 现代框架中的克隆实践

Vue 状态管理:
// 使用深克隆创建初始状态
const initialState = deepClone(fetchedData);// 在组件中使用
export default {data() {return {localState: deepClone(this.initialState)}}
}
React 状态更新:
function reducer(state, action) {switch (action.type) {case 'UPDATE_USER':// 深合并示例return {...state,user: {...state.user,...action.payload}};case 'ADD_ITEM':// 数组深克隆return {...state,items: [...state.items, action.item]};}
}

六、克隆方案选择指南

场景特征推荐方案原因说明
简单对象无嵌套浅克隆性能最优
需要保留特殊对象类型structuredClone()原生支持多种类型
包含函数/循环引用lodash.cloneDeep完整处理复杂情况
现代浏览器环境structuredClone()原生性能最好
需要处理DOM节点自定义实现需特殊处理DOM API
超大对象(>10MB)分块克隆避免阻塞主线程
高频克隆操作(>1000次/秒)浅克隆+immutable性能敏感场景优化

七、总结与最佳实践

  1. 理解数据本质:区分基本类型和引用类型
  2. 明确克隆需求:深克隆还是浅克隆
  3. 选择合适方案
    • 优先使用浏览器原生API:structuredClone()
    • 复杂场景使用成熟库:lodash的_.cloneDeep()
    • 特殊需求自定义实现
  4. 处理边界情况
    • 循环引用使用WeakMap
    • 特殊对象类型单独处理
    • Symbol属性不能忽略
  5. 性能考量
    • 避免在循环中深度克隆大对象
    • 考虑使用不可变数据结构
    • 超大对象使用分块处理
// 最佳实践示例
import { cloneDeep } from 'lodash-es';// 需要完全隔离的数据
const configBackup = cloneDeep(runtimeConfig);// 浏览器环境简单克隆
const stateSnapshot = structuredClone(appState);// 浅克隆配置对象
const currentSettings = { ...defaultSettings, ...userSettings };

掌握深浅克隆的区别和实现方式,是写出健壮JavaScript应用的关键技能。根据具体场景选择合适的克隆策略,可以有效避免数据污染和意外行为。

相关文章:

ES6 深克隆与浅克隆详解:原理、实现与应用场景

ES6 深克隆与浅克隆详解:原理、实现与应用场景 一、克隆的本质与必要性 在 JavaScript 中,数据分为两大类型: 基本类型:Number、String、Boolean、null、undefined、Symbol、BigInt引用类型:Object、Array、Functio…...

Go Gin框架深度解析:高性能Web开发实践

Go Gin框架深度解析:高性能Web开发实践 Gin框架核心特性概览 Gin是用Go语言编写的高性能Web框架,以其​​闪电般的路由性能​​(基于httprouter)和​​极简的API设计​​著称: package mainimport "github.com…...

mybatis 参数绑定错误示范(1)

采用xml形式的mybatis 错误示例&#xff1a; server伪代码为&#xff1a; Map<String, Object> findMapNew MapUtil.<String, Object>builder().put("applyUnit", appUnit).put("planYear", year ! null ? year : -1).put("code&quo…...

每天掌握一个Linux命令 - rpm

Linux 命令工具 rpm 使用指南 Linux 命令工具 rpm 使用指南一、工具概述二、安装方式1. 系统预装2. 源码编译安装&#xff08;极少场景&#xff09; 三、核心功能四、基础用法1. 安装软件包2. 升级软件包3. 查询软件包信息4. 卸载软件包5. 验证文件完整性 五、进阶操作1. 批量操…...

常见的MySQL索引类型

在MySQL中&#xff0c;索引是用来提高数据库查询效率的一种数据结构。根据不同的使用场景和需求&#xff0c;MySQL提供了多种类型的索引&#xff0c;每种索引都有其特定的应用场景和优化效果。下面是一些常见的MySQL索引类型&#xff1a; 1. B-Tree索引&#xff1a; 这是最常…...

01串(二进制串)与集合之间存在天然的对应关系 ← bitset

【集合的二进制表示‌】 ● 01 串&#xff08;二进制串&#xff09;与集合之间存在天然的对应关系。对应机理为每个二进制位可以表示集合中一个元素的存在&#xff08;1&#xff09;或不存在&#xff08;0&#xff09;。例如&#xff0c;集合 {a, b, c} 的子集 {a, c} 可以表示…...

153页PPT麦肯锡咨询流程管理及企业五年发展布局构想与路径规划

麦肯锡咨询的流程管理以其高度结构化、数据驱动和结果导向的核心特点著称&#xff0c;旨在为客户提供清晰、可行且价值最大化的解决方案。其典型流程可概括为以下几个关键阶段&#xff1a;下载资料请查看文章中图片右下角信息 问题界定与结构化&#xff1a; 这是流程的基石。麦…...

[特殊字符] 革命性AI提示词优化平台正式开源!

AI时代最强大的Prompt工程师已经到来&#xff01; 你是否还在为写不出高质量提示词而头疼&#xff1f;是否羡慕那些能够驾驭AI、让ChatGPT、Claude乖乖听话的"提示词大师"&#xff1f;今天&#xff0c;我们为你带来一个颠覆性的解决方案——TokenAI Auto-Prompt&…...

我的概要设计模板(以图书管理系统为例)

一、总述 1.1 需求或目标 随着数字化阅读普及&#xff0c;传统图书馆管理方式效率低下、资源检索不便。为提升图书管理效率&#xff0c;方便读者借阅与查询&#xff0c;公司计划开发 “在线图书管理系统”&#xff0c;实现图书的电子化管理、快速检索、在线借阅等功能&#x…...

【使用】【经验】docker 清理未使用的镜像的命令

docker images prune在 Docker 中清理未使用的镜像&#xff08;包括悬空镜像和完全未被引用的镜像&#xff09;&#xff0c;可以使用以下命令&#xff1a; 1. ​删除所有悬空镜像​&#xff08;推荐常用&#xff09; docker image prune​悬空镜像 (dangling images)​​ 是指…...

DrissionPage爬虫包实战分享

一、爬虫 1.1 爬虫解释 爬虫简单的说就是模拟人的浏览器行为&#xff0c;简单的爬虫是request请求网页信息&#xff0c;然后对html数据进行解析得到自己需要的数据信息保存在本地。 1.2 爬虫的思路 # 1.发送请求 # 2.获取数据 # 3.解析数据 # 4.保存数据 1.3 爬虫工具 Dris…...

iptables实战案例

目录 一、实验拓扑 二、网络规划 三、实验要求 四、环境准备 1.firewall &#xff08;1&#xff09;配置防火墙各大网卡IP并禁用 firewall和selinux &#xff08;2&#xff09;打开firewall路由转发 2.PC1&#xff08;内网&#xff09; &#xff08;1&#xff09;配置防…...

机器学习与深度学习07-随机森林01

目录 前文回顾1.随机森林的定义2.随机森林中的过拟合3.随机森林VS单一决策树4.随机森林的随机性 前文回顾 上一篇文章链接&#xff1a;地址 1.随机森林的定义 随机森林&#xff08;Random Forest&#xff09;是一种集成学习算法&#xff0c;用于解决分类和回归问题。它基于决…...

回归分析-非线性回归及岭回归.docx

一.题目要求1.用SPSS软件练习建立多元线性回归方程,分析数据的多重共线性,利用后退法和逐步回归法选择变量,练习用岭回归方法处理该模型数据并作比较 2.用SPSS软件练习建立模型的非线性回归方程 二.数据分析(一)题目:课本7.6 1、数据 一家大型商业银行有多家分行,近年来…...

Google AI 模式下的SEO革命:生成式搜索优化(GEO)与未来营销策略

一、搜索范式转变&#xff1a;从链接引导到答案交付 Google自2023年起逐步推出AI搜索功能&#xff0c;经历了SGE&#xff08;Search Generative Experience&#xff09;和Gemini阶段&#xff0c;最终在2025年全面上线了「AI Mode」搜索模式。与此同时&#xff0c;也保留了一种过…...

docker创建postgreSql带多个init的sql

好的&#xff01;下面是一个完整的可运行项目结构&#xff0c;包含&#xff1a; ✅ docker-compose.yml&#xff1a;启动 PostgreSQL&#xff08;支持 pgvector&#xff09; ✅ init-db.sql&#xff1a;创建数据库 myapp ✅ init-schema.sql&#xff1a;在 myapp 中建表并初始…...

掌握 MotionLayout:交互动画开发

前言 在 Android 开发中&#xff0c;系统自带的属性动画&#xff08;如 ObjectAnimator 和 ValueAnimator&#xff09;虽然功能强大&#xff0c;但在复杂动画场景下&#xff0c;第三方动画库能提供更高效的开发体验和更丰富的效果。本文将深入解析 Lottie、MotionLayout、Andr…...

SpringBoot中缓存@Cacheable出错

SpringBoot中使用Cacheable: 错误代码&#xff1a; Cacheable(value "FrontAdvertiseVOList", keyGenerator "cacheKey") Override public List<FrontAdvertiseVO> getFrontAdvertiseVOList(Integer count) {return this.list(Wrappers.<Adve…...

iOS UIActivityViewController 组头处理

0x00 情形一 - (void)shareAction1 {// 当前 View 转成图片UIImage *image [self snapshotImage:self.view];NSArray *activityItems [image];UIActivityViewController *activityVC [[UIActivityViewController alloc] initWithActivityItems:activityItems applicationAc…...

分布式电源接入配电网的自适应电流保护系统设计与实现

分布式电源接入配电网的自适应电流保护系统设计与实现 一、引言 随着可再生能源的快速发展,分布式电源(Distributed Generation, DG)大规模接入配电网,传统保护系统面临以下挑战: 潮流方向改变导致保护误动/拒动故障电流水平波动影响保护灵敏度多类型故障(单相/两相/三…...

鸿蒙版Taro 搭建开发环境

鸿蒙版Taro 搭建开发环境 一、配置鸿蒙环境 下载安装 DevEco 建议使用最新版本的 IDE&#xff0c;当前为 5.0.5Release 版本。 二、创建鸿蒙项目 打开 DevEco&#xff0c;点击右上角的 Create Project&#xff0c;在 Application 处选择 Empty Ability&#xff0c;点击 Ne…...

论对生产力决定生产关系的批判:突破决定论的桎梏

笔言: 在学生时代认为"生产力决定生产关系"很有道理&#xff0c;但是进入社会参与市场竞争时候&#xff0c;才发现这种想法太天真了&#xff0c;当生产力一只赔钱时候谁也不会感兴趣&#xff1b;当生产力产生利润&#xff0c;比如1%30%&#xff0c;100%&#xff0c;3…...

ESOP交易系统搭建全景指南:从合规基石到价值跃迁

第一章 重新定义ESOP&#xff1a;合规性与流动性的平衡艺术 1.1 ESOP的本质演进 传统认知误区&#xff1a;员工持股计划股权分配工具 现代定义&#xff1a; ESOP是企业资本运作的中枢神经系统&#xff0c;贯穿“激励授予→行权管理→减持流通→市值协同”全链条&#xff0c;需…...

GICv3电源管理

在符合GICv3体系结构的实现中&#xff0c;CPU接口和PE必须位于相同的电源域&#xff0c;但这不必与关联的Redistributor所在的电源域相同。 这意味着可能会出现PE及其CPU interface断电&#xff0c;而Redistributor、Distributor和its上电的情况。在这种情况下&#xff0c;GIC架…...

《TCP/IP 详解 卷1:协议》第3章:链路层

以太网和IEEE802局域网/城域网标准 IEEE802局域网/城域网标准 IEEE 802 是一组由 IEEE&#xff08;电气与电子工程师协会&#xff09;定义的局域网和城域网通信标准系列&#xff0c;涵盖了从物理层到链路层的多个网络技术。其中&#xff1a; IEEE 802.3 定义的是传统的以太网…...

centos 9/ubuntu 一次性的定时关机

方法一 # 15 表示15分钟以后自动关机 sudo shutdown -h 15方法二&#xff1a; sudo dnf install at -y # 晚上十点半关机 echo "shutdown -h now" | at 22:30 # 检查是否设置成功命令 atq [rootdemo-192 ~]# atq 1 Wed Jun 4 11:12:00 2025 a root # 取消定时计划…...

Elasticsearch从安装到实战、kibana安装以及自定义IK分词器/集成整合SpringBoot详细的教程(二)

package com.test.xulk.es.entity.mapper;import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.test.xulk.es.entity.Hotel;public interface HotelMapper extends BaseMapper<Hotel> { }集成Springboot 项目里面 官方地址&#xff1a; Elasticsearch …...

Java自动类型转换的妙用

Java中的自动类型转换&#xff08;也称为隐式类型转换&#xff09;是指在不需要显式指定转换的情况下&#xff0c;Java编译器自动将一种数据类型转换为另一种数据类型。这种特性在编程中有许多妙用&#xff0c;以下是一些常见的应用场景和优点&#xff1a; 1. 简化代码 自动类…...

数据库管理-第333期 Oracle 23ai:RAC打补丁完全不用停机(20250604)

数据库管理333期 2025-06-04 数据库管理-第333期 Oracle 23ai&#xff1a;RAC打补丁完全不用停机&#xff08;20250604&#xff09;1 概念2 要求3 操作流程4 转移失败处理总结 数据库管理-第333期 Oracle 23ai&#xff1a;RAC打补丁完全不用停机&#xff08;20250604&#xff0…...

【DAY39】图像数据与显存

内容来自浙大疏锦行python打卡训练营 浙大疏锦行 知识点&#xff1a; 图像数据的格式&#xff1a;灰度和彩色数据模型的定义显存占用的4种地方 模型参数梯度参数优化器参数数据批量所占显存神经元输出中间状态 batchisize和训练的关系 作业&#xff1a;今日代码较少&#xff0…...