前端中的深拷贝

第1部分:引言

深拷贝:前端开发的隐形守护者

在前端开发的世界里,数据的传递和状态的管理是构建用户界面的基础。然而,数据的复制常常被忽视,直到它引发bug,我们才意识到它的重要性。深拷贝,这个看似简单的概念,实际上是前端开发中一个关键的环节,它确保了数据的独立性和安全性。

1.1 为什么深拷贝如此重要?

在JavaScript中,数据类型分为原始数据类型引用数据类型。对于原始数据类型(如数字、字符串、布尔值等),变量直接存储值,因此复制操作是安全的。但对于引用数据类型(如数组、对象等),变量存储的是指向内存中对象的引用。在这种情况下,简单的赋值或浅拷贝会导致原始数据和复制数据指向同一内存地址,任何对复制数据的修改都会反映到原始数据上,这可能不是我们想要的结果。

1.2 浅拷贝的局限性

浅拷贝,如使用赋值操作符或展开运算符,虽然在某些情况下足够用,但它无法创建对象内部属性的副本。这意味着如果对象内部包含了对象或数组,浅拷贝只会复制引用,而不是实际的数据。这在处理嵌套数据结构时尤其危险,因为它可能导致不可预见的副作用。

1.3 深拷贝的必要性

深拷贝,顾名思义,会递归地复制对象的所有属性,无论这些属性是基本类型还是复杂的引用类型。这样,原始数据和复制数据完全独立,互不影响。在开发复杂的应用程序时,如状态管理、组件属性传递、函数参数传递等场景,深拷贝提供了一种安全的数据操作方式。

1.4 深拷贝的应用场景
  • 状态管理:在Redux或Vuex等状态管理库中,深拷贝确保了状态的不可变性,避免了直接修改状态导致的不可预测行为。
  • 组件开发:在React或Vue中,组件的props或state可能需要复制以避免父子组件间的数据污染。
  • 函数编程:在函数式编程中,不可变性是核心原则之一,深拷贝是实现这一原则的关键技术。

第2部分:基础知识

2.1 浅拷贝与深拷贝的定义

在深入讨论之前,我们需要明确两个基本概念:浅拷贝深拷贝

  • 浅拷贝:创建一个新对象,但是这个对象的属性值是指向原始对象属性值的引用。如果原始对象的属性是基本类型,那么浅拷贝会复制这些值;如果属性是引用类型,那么新对象会复制引用,而不是引用的对象本身。

  • 深拷贝:创建一个新对象,并且递归地复制这个对象的所有子对象,直到所有的属性都是独立的副本。这意味着原始对象和复制对象之间不存在引用共享。

2.2 浅拷贝与深拷贝的示例对比

为了更直观地理解两者的区别,让我们通过一些JavaScript代码示例来展示。

浅拷贝示例:

let originalObj = { a: 1, b: { c: 2 } };
let shallowCopy = { ...originalObj }; // 使用展开运算符进行浅拷贝console.log(shallowCopy); // { a: 1, b: { c: 2 } }// 修改浅拷贝对象的引用属性
shallowCopy.b.c = 3;// 原始对象也发生了变化
console.log(originalObj); // { a: 1, b: { c: 3 } }

深拷贝示例:

let originalObj = { a: 1, b: { c: 2 } };
let deepCopy = JSON.parse(JSON.stringify(originalObj)); // 使用JSON方法进行深拷贝console.log(deepCopy); // { a: 1, b: { c: 2 } }// 修改深拷贝对象的引用属性
deepCopy.b.c = 3;// 原始对象保持不变
console.log(originalObj); // { a: 1, b: { c: 2 } }
2.3 浅拷贝的常见方法
  • 使用赋值操作符(=)。
  • 使用数组的slice()方法。
  • 使用展开运算符(...)。
  • 使用Object.assign()方法。

示例:

// 使用Object.assign()进行浅拷贝
let original = { a: 1, b: 2 };
let shallowCopy = Object.assign({}, original);
original.b = 3; // 改变原始对象的属性值
console.log(shallowCopy); // { a: 1, b: 2 } 浅拷贝对象不受影响
2.4 深拷贝的挑战

尽管深拷贝提供了数据独立性,但它也带来了一些挑战:

  • 性能问题:深拷贝可能涉及到大量的内存分配和数据复制,这在处理大型或复杂的数据结构时可能会影响性能。
  • 循环引用:对象如果包含循环引用,标准的深拷贝方法可能无法正确处理。
  • 特殊对象:如Date、RegExp、Function等特殊对象在深拷贝时可能需要特别处理。
2.5 深拷贝的实现方法

实现深拷贝的方法有很多,包括但不限于:

  • 使用递归函数。
  • 使用序列化和反序列化(如JSON方法)。
  • 使用第三方库,如lodash的_.cloneDeep()

示例:

// 使用递归函数实现深拷贝
function deepCopy(obj, hash = new WeakMap()) {if (obj === null) return null;if (typeof obj !== 'object') return obj;if (hash.has(obj)) return hash.get(obj);let result = new obj.constructor();hash.set(obj, result);for (let key of Object.keys(obj)) {result[key] = deepCopy(obj[key], hash);}return result;
}let originalObj = { a: 1, b: { c: 2 } };
let customDeepCopy = deepCopy(originalObj);
console.log(customDeepCopy); // { a: 1, b: { c: 2 } }

第3部分:JavaScript中的拷贝机制

3.1 原始数据类型与引用数据类型的拷贝

JavaScript中的数据类型分为原始数据类型(Primitive Data Types)和引用数据类型(Reference Data Types)。理解这两种数据类型的拷贝机制对于掌握深拷贝至关重要。

  • 原始数据类型:包括NumberStringBooleanUndefinedNull。这些类型的数据是按值访问的,所以拷贝操作会创建原始值的一个全新副本。

  • 引用数据类型:包括ObjectArrayFunction等。这些类型的数据是按引用访问的,拷贝操作创建的是引用的副本,而不是实际对象的副本。

示例:

let originalPrimitive = 42;
let copiedPrimitive = originalPrimitive;
originalPrimitive = 24;console.log(copiedPrimitive); // 42,原始值被复制let originalObject = { a: 1 };
let copiedObject = originalObject;
copiedObject.a = 2;console.log(originalObject); // { a: 2 },引用被复制,修改影响了原始对象
3.2 常见的浅拷贝方法

浅拷贝方法适用于复制对象的第一层属性,但不会递归复制嵌套对象。

  • 赋值操作符:创建原始对象的引用副本。
  • 扩展运算符...,用于数组和对象字面量。
  • Object.assign():用于对象,将源对象的所有可枚举属性复制到目标对象。
  • 数组的slice()方法:仅适用于数组。

示例:

let originalArray = [1, 2, 3];
let copiedArray = originalArray.slice(); // [1, 2, 3]originalArray.push(4);
console.log(copiedArray); // [1, 2, 3],数组的slice()方法创建了新数组
3.3 浅拷贝的局限性

浅拷贝虽然简单易用,但它的局限性在于无法处理嵌套的对象或数组。此外,它也不能复制特殊对象,如函数、日期等。

示例:

let original = { a: 1, b: { c: 2 } };
let copied = { ...original };copied.b.c = 3; // 这将影响原始对象的b.c属性console.log(original); // { a: 1, b: { c: 3 } }
3.4 深拷贝的实现策略

深拷贝需要递归地复制对象的所有属性,包括嵌套的对象和数组。

  • 递归拷贝:手动实现深拷贝函数,递归地复制所有属性。
  • 序列化与反序列化:使用JSON.stringify()JSON.parse(),但有局限性。
  • 第三方库:如lodash的_.cloneDeep()

示例:

function deepCopy(obj) {if (obj === null || typeof obj !== 'object') {return obj;}let temp = Array.isArray(obj) ? [] : {};for (let key in obj) {temp[key] = deepCopy(obj[key]);}return temp;
}let original = { a: 1, b: { c: 2 } };
let copied = deepCopy(original);copied.b.c = 3;
console.log(original); // { a: 1, b: { c: 2 } },原始对象未受影响
3.5 特殊对象的拷贝

特殊对象如函数、日期、正则表达式等,需要特别注意,因为它们可能包含方法或状态,这些不能通过简单的复制来克隆。

示例:

let originalDate = new Date();
let copiedDate = new Date(originalDate);console.log(copiedDate.getTime() === originalDate.getTime()); // true,但它们是不同的实例

第4部分:深拷贝的挑战

4.1 循环引用问题

深拷贝面临的一个主要挑战是处理对象中的循环引用。循环引用发生在对象直接或间接地引用自己时。标准的深拷贝实现可能无法处理这种情况,导致无限递归或错误。

示例:

let obj = {};
obj.self = obj; // 创建循环引用function deepCopy(obj, hash = new WeakMap()) {if (obj === null || typeof obj !== 'object') return obj;if (hash.has(obj)) return hash.get(obj); // 检查循环引用let copy = Array.isArray(obj) ? [] : {};hash.set(obj, copy);for (let key of Object.keys(obj)) {copy[key] = deepCopy(obj[key], hash);}return copy;
}let newObj = deepCopy(obj);
console.log(newObj.self === newObj); // true,成功处理循环引用
4.2 不同类型数据的拷贝问题

深拷贝需要能够处理各种类型的数据,包括但不限于普通对象、数组、函数、日期、正则表达式等。每种数据类型都有其特殊性,需要特别处理。

示例:

let original = {number: 42,string: 'Hello',boolean: true,null: null,undefined: undefined,date: new Date(),regexp: /test/,func: function() { console.log('Function'); }
};function deepCopyWithTypes(obj) {if (obj === null) return null;if (obj instanceof Date) return new Date(obj);if (obj instanceof RegExp) return new RegExp(obj.source, obj.flags);if (typeof obj === 'function') return obj; // 函数通常不复制let copy = Array.isArray(obj) ? [] : {};for (let key of Object.keys(obj)) {copy[key] = deepCopyWithTypes(obj[key]);}return copy;
}let copied = deepCopyWithTypes(original);
console.log(copied.date instanceof Date); // true
console.log(copied.regexp instanceof RegExp); // true
4.3 性能问题

深拷贝可能会消耗大量计算资源,尤其是在处理大型或复杂的数据结构时。性能问题可能会影响到应用程序的响应速度和用户体验。

示例:

let largeObj = {};
for (let i = 0; i < 10000; i++) {largeObj['key' + i] = 'value' + i;
}console.time('deepCopy');
deepCopy(largeObj);
console.timeEnd('deepCopy'); // 测量深拷贝所需的时间
4.4 不可枚举属性和Symbol属性

JavaScript对象可能包含不可枚举属性或使用Symbol作为键的属性。这些属性可能不会被标准的深拷贝方法复制。

示例:

let original = {enumerable: 'This is enumerable',[Symbol('key')]: 'This is a Symbol key'
};Object.defineProperty(original, 'nonEnumerable', {value: 'This is non-enumerable',enumerable: false
});let copied = JSON.parse(JSON.stringify(original));
console.log(copied.enumerable); // "This is enumerable"
console.log(copied[Symbol('key')]); // undefined,Symbol属性未被复制
console.log(copied.nonEnumerable); // undefined,不可枚举属性未被复制
4.5 深拷贝的陷阱

深拷贝可能看起来简单,但实际上存在许多陷阱,如原型链的拷贝、特殊对象的处理等。开发者需要对这些潜在问题有所了解,并在实现深拷贝时加以考虑。

示例:

function Person(name) {this.name = name;
}Person.prototype.greet = function() {console.log(`Hello, my name is ${this.name}`);
};let person1 = new Person('Alice');
let person2 = deepCopy(person1); // 使用前面定义的deepCopy函数person2.greet(); // 错误:person2.greet is not a function

看到这,欢迎友友们关注我的公众号:行动圆周率
或扫描关注

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/359635.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Springboot 共享电动单车管理系统-计算机毕业设计源码08401

目 录 摘要 1 绪论 1.1背景及意义 1.2国内外研究概况 1.3研究方法 1.4论文结构与章节安排 2 共享电动单车管理系统系统分析 2.1 可行性分析 2.1.1 技术可行性分析 2.1.2 经济可行性分析 2.1.3 法律可行性分析 2.2 系统功能分析 2.2.1 功能性分析 2.2.2 非功能性分…

React Suspense的原理

React Suspense组件的作用是当组件未完成加载时&#xff0c;显示 fallback 组件。那么 Suspense 是如何实现的呢&#xff1f;React 的渲染是通过 Fiber 进行的&#xff0c;Suspense 的更新机制也是要围绕 Fiber 架构进行的。Suspense 是由两部分组成&#xff0c;实际 UI 子组件…

自学指南:必备书籍清单--近100本R语言及生物信息相关书籍

R语言是一种功能丰富的编程语言&#xff0c;数据处理、统计分析是大家所熟知的基本功能。开源免费、活跃的全球社区、灵活可扩展等优点促使R语言飞速发展。目前&#xff0c;CRAN 软件包存储库包含 20446 个可用软件包&#xff0c;涵盖了从生物信息到金融分析等广泛的应用领域。…

spring-boot-devtools热部署功能集成使用

0.参考其他文章 Spring Boot之Spring-devtools热部署&#xff1a;实现快速开发与调试-阿里云开发者社区关于IDEA2022开启热部署没有compiler.automake.allow.when.app.running的解决方案-CSDN博客 1. Spring DevTools简介 Spring DevTools是Spring团队开发的一个模块&#xf…

理解人体手臂七个自由度对应的运动

写本篇的目的在于&#xff0c;我发现很多人理不清人体手臂运动时内收/外展、屈曲/伸展等动作描述的关系&#xff0c;包括我自己也是&#xff01; 我每次要用到的时候都要去查&#xff0c;记不下来&#xff0c;比较麻烦&#xff0c;于是归纳本篇&#xff0c;包含了我本人的理解 …

解析PDF文件中的图片为文本

解析PDF文件中的图片为文本 1 介绍 解析PDF文件中的图片&#xff0c;由两种思路&#xff0c;一种是自己读取PDF文件中的图片&#xff0c;然后用OCR解析&#xff0c;例如&#xff1a;使用PyMuPDF读取pdf文件&#xff0c;再用PaddleOCR或者Tesseract-OCR识别文字。另一种使用第…

Vscode中的行尾序列CRLF/LF不兼容问题

最近开发的的时候&#xff0c;打开项目文件经常会出现爆红错误提示信息&#xff0c;显示如下图&#xff1a; 这东西太烦人了&#xff0c;毕竟谁都不希望在遍地都是爆红的代码里写东西&#xff0c;就像能解决这个问题&#xff0c;根据提示可以知道这是vscode中使用的prettier插件…

图片转pdf,图片转pdf在线转换,在线图片转pdf

图片转PDF&#xff0c;听起来似乎是一个简单的操作&#xff0c;但实际上&#xff0c;它涉及到许多细节和技巧。有时候我们需要将图片转换为PDF格式&#xff0c;以便于分享、打印或保存。那么&#xff0c;如何将图片转换成PDF呢&#xff1f;接下来&#xff0c;我将为您详细介绍几…

Vue3中的常见组件通信(超详细版)

Vue3中的常见组件通信 概述 ​ 在vue3中常见的组件通信有props、mitt、v-model、 r e f s 、 refs、 refs、parent、provide、inject、pinia、slot等。不同的组件关系用不同的传递方式。常见的撘配形式如下表所示。 组件关系传递方式父传子1. props2. v-model3. $refs4. 默认…

零基础MySQL完整学习笔记

零基础MySQL完整学习笔记 1. 基础操作(必须会!)1.1 修改密码(4种方法)1.2 创建新用户1.3 导入数据库 2. SQL四种语言介绍2.1 DDL(数据库定义语言)2.2 DML(数据操纵语言)2.3 DCL(数据库控制语言)2.4 TCL(事务控制语言) 3. 数据库操作3.1 创建数据库3.2 查询数据库3.3 删除数据库…

【ArcGIS微课1000例】0120:ArcGIS批量修改符号的样式(轮廓)

ArcGIS可以批量修改符号的样式,如样式、填充颜色、轮廓等等。 文章目录 一、加载实验数据二、土地利用符号化三、批量修改符号样式四、注意事项一、加载实验数据 订阅专栏后,从私信查收专栏配套的完整实验数据包,打开0120.rar中的土地利用数据,如下图所示: 查看属性表: …

SpringMVC系列十: 中文乱码处理与JSON处理

文章目录 中文乱码处理自定义中文乱码过滤器Spring提供的过滤器处理中文 处理json和HttpMessageConverter<T>处理JSON-ResponseBody处理JSON-RequestBody处理JSON-注意事项和细节HttpMessageConverter<T\>文件下载-ResponseEntity<T\>作业布置 上一讲, 我们学…

python判断语句

目录 布尔类型和比较运算符if语句的基本格式if else 语句if elif else 语句判断语句的嵌套 布尔类型和比较运算符 1、布尔类型 bool布尔类型只有两个结果&#xff1a;真或假 布尔类型的字面量&#xff1a; True 表示真&#xff08;是、肯定&#xff09; False 表示假&#x…

【论文精读】ViM: Out-Of-Distribution with Virtual-logit Matching 使用虚拟分对数匹配的分布外检测

文章目录 一、文章概览&#xff08;一&#xff09;问题来源&#xff08;二&#xff09;文章的主要工作&#xff08;三&#xff09;相关研究 二、动机&#xff1a;Logits 中缺失的信息&#xff08;一&#xff09;logits&#xff08;三&#xff09;基于零空间的 OOD 评分&#xf…

[leetcode]add-strings 字符串相加

. - 力扣&#xff08;LeetCode&#xff09; class Solution { public:string addStrings(string num1, string num2) {int i num1.length() - 1, j num2.length() - 1, add 0;string ans "";while (i > 0 || j > 0 || add ! 0) {int x i > 0 ? num1[i…

msvcp120.dll丢失怎么办,找不到msvcp120.dll的多种解决方法

最近&#xff0c;我在运行一个程序时遇到了一个错误&#xff0c;系统提示找不到msvcp120.dll文件&#xff0c;无法继续执行代码。这让我感到非常困扰&#xff0c;因为这个问题导致我无法正常运行这个程序。经过一番搜索和尝试&#xff0c;我找到了几种修复这个问题的方法&#…

pdf转图片转换器,pdf转图片的工具

在日常的工作和学习中&#xff0c;我们经常会遇到需要将PDF文件转换为图片格式的情况。那么&#xff0c;如何才能将PDF格式转换为图片格式呢&#xff1f;今天&#xff0c;我将为大家介绍几种简单易用的方法&#xff0c;帮助大家轻松实现PDF转图片。 打开“轻云pdf处理官网网站”…

WEB界面上使用ChatGPT

&#xff08;作者&#xff1a;陈玓玏&#xff09; 开源项目&#xff0c;欢迎star哦&#xff0c;https://github.com/tencentmusic/cube-studio 随着大模型不断发展&#xff0c;现在无论写代码&#xff0c;做设计&#xff0c;甚至老师备课、评卷都可以通过AI大模型来实现了&…

后端实现预览pdf,mp4,图片

PDF预览 /*** pdf预览* param response*/RequestMapping(value "/preview")public void showPdf(HttpServletResponse response) {try {//String filePath this.getClass().getClassLoader().getResource("../../static/pdf/readme.pdf").getPath();Stri…

线程池概念、线程池的不同创建方式、线程池的拒绝策略

文章目录 &#x1f490;线程池概念以及什么是工厂模式&#x1f490;标准库中的线程池&#x1f490;什么是工厂模式&#xff1f;&#x1f490;ThreadPoolExecutor&#x1f490;模拟实现线程池 &#x1f490;线程池概念以及什么是工厂模式 线程的诞生是因为&#xff0c;频繁的创…