前端中的深拷贝

news2024/11/17 23:47:42

第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.coloradmin.cn/o/1858834.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

NC--介绍-未加密加密后-流量抓包对比

免责声明:本节仅做技术交流与学习... 目录 介绍: 用法: 未加密--流量抓包 加密: 攻击端 靶机 抓包分析: 介绍: nc 是一个Linux环境下常用的工具命令&#xff0c;可以用来帮助开发者查询和解决网路问题&#xff0c;通常被认为是 NetCat 工具的缩写&#xff0c;在网络工具…

arduino按钮

Arduino - Button Arduino - 按钮 参考&#xff1a; ezButton-按钮库从按钮开关看上拉pull-up电阻、下拉电阻按键的防抖动处理 The button is also called pushbutton, tactile button or momentary switch. It is a basic component and widely used in many Arduino projec…

06 - matlab m_map地学绘图工具基础函数 - 绘制海岸线

06 - matlab m_map地学绘图工具基础函数 - 绘制海岸线 0. 引言1. 关于m_coast2. 关于m_gshhs3. 关于m_gshhs_c、m_gshhs_I、m_gshhs_i、m_gshhs_h、m_gshhs_f4. 关于m_shaperead5. 结语 0. 引言 本篇介绍下m_map中添加绘制海岸线的一系列函数及其用法&#xff0c;主要函数包括m…

分享一个解决 EF 性能低的思路,通过 Python 访问心跳侦测 API 保持 EF 在线

前言 .NET FrameWork EF&#xff08;Entity Framework&#xff09;是一个很优秀的 ORM 框架&#xff0c;对于提高工作效率很有帮助&#xff0c;但由于底层原因&#xff0c;在程序启动时&#xff0c;EF 需要初始化和加载模型及其对应的数据库元数据等等&#xff0c;涉及到数据库…

分布式数据库垂直分片如何计算?

在复习分布式数据库这一部分时&#xff0c;发现垂直分片较难理解&#xff0c;上网上查找资料&#xff0c;也发现找不到&#xff0c;固在考试结束后将自己的理解记录下来。 垂直分片的计算总体上分为三个步骤;1、属性亲和度矩阵AA的计算&#xff1b;2、聚类亲和度CA矩阵的计算&a…

“了解MySQL中的enum枚举数据类型“

目录 # 开篇 1. 创建包含枚举类型的表 2. 插入枚举类型的数据 3. 查询包含枚举类型的表 4. 更新枚举类型的数据 5. 使用枚举类型的好处 注意事项 示例总结 附加 # 开篇 在数据库中&#xff0c;枚举&#xff08;ENUM&#xff09;是一种数据类型&#xff0c;用于存储一组…

vue3 运用高德地图 自定义弹框 为信息窗体 添加 new AMaps.value.InfoWindow 添加事件

效果图 划过散点的时候出现每个三点位置的数据提示 点击具体散点获取展示信息弹框&#xff0c;并为其添加点击事件 注意点&#xff1a; 1 即使是用的vue&#xff0c;也不能使用click为窗体添加点击事件&#xff0c;需要使用onclick&#xff0c; &#xff08;原因&#xff1a…

Android 13 创建静态快捷方式shortcuts

参考 创建快捷方式 原生系统上&#xff0c;长按应用图标显示快捷方式&#xff0c;点击快捷方式就打开应用的某个页面。 给自己的应用也加一下。 1.清单文件添加 在应用的主页面添加如下&#xff0c;shortcuts 就是要配置的文件。 <meta-dataandroid:name"android.…

01 Swift 基础语法(变量,常量,注释,整数,浮点数,元组)

Swift 介绍 Swift 是一门用于开发 iOS、macOS、watchOS 和 tvOS 应用的新编程语言。它以安全、快速和互动的特点而著称。Swift 提供了代码预览&#xff08;playgrounds&#xff09;功能&#xff0c;使程序员可以在不编译和运行应用程序的情况下&#xff0c;实时运行 Swift 代码…

多模态-大模型:MLLM综述(适用初学)

文章目录 前言一、多模态模型基础知识二、多模态指令调优&#xff08;M-IT&#xff09;1.MLLM基础2.模态对齐3.数据获取4.模态桥接 三、多模态上下文学习&#xff08;M-ICL&#xff09;三、多模态思维链 (M-CoT)四、LLM辅助视觉推理1.训练范式2. LLM功能 五、一些思考总结 前言…

索引的分类和回表查询——Java全栈知识(29)

索引的分类和回表查询 Mysql 的索引按照类型可以分为以下几类&#xff0c;但是我们使用的 InnoDB 只支持主键索引&#xff0c;唯一索引&#xff0c;普通索引&#xff0c;并不支持全文索引。 1、聚集索引和二级索引 InnoDB 可以将索引分为两类分别是聚集索引和二级索引&…

模板匹配算法:基于模板相关性匹配的手写数字识别

1 前言 得益于硬件技术的发展&#xff0c;基于深度学习的各种识别方法如火如荼&#xff0c;在各种应用场景中都取得很好的效果。本人入行深度学习领域若干年&#xff0c;做过很多项目的工程化评估&#xff0c;对于神经网络是如何工作的也解释不清楚&#xff0c;只是知道这样做是…

如何借用物联网快速实现高标准农田信息化

如何借用物联网快速实现高标准农田信息化 高标准农田信息化&#xff0c;作为现代农业发展的重要基石&#xff0c;是指在建设高产、稳产、节水、环保的农田基础上&#xff0c;深度融合现代信息技术&#xff0c;实现农田管理的精准化、智能化和高效化。物联网&#xff08;Intern…

[Angew]:调整单原子 Pt1/CeO2催化剂中铂的局部环境以实现稳定的低温 CO 氧化

通过热冲击合成调整孤立 Pt2+的局部环境,开发出一种用于 CO 氧化的高活性和坚固的单原子 Pt1/CeO2催化剂(见图)。惰性气氛中的超快冲击波在非对称 Pt1O4结构中产生 Pt 单原子,从而大大增强了低温活性,并且在氧化条件下仍能保持活性。 摘要:通过原子捕获(AT,800C空气中…

pycharm在pytorch环境的使用

前言 有时我们使用jupyter感觉不习惯&#xff0c;想使用编译器进行相关任务。在安装好pytorch环境下&#xff0c;我们可以使用pycharm等编译器进行完成任务等操作。 安装pycharm不做赘述。配置安装好pytorch环境可以参考我前一篇博客&#xff1a; 深度学习工具jupyter创建并检…

基于Java考研助手网站设计和实现(源码+LW+调试文档+讲解等)

&#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN作者、博客专家、全栈领域优质创作者&#xff0c;博客之星、平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f31f;文末获取源码数据库&#x1f31f; 感兴趣的可以先收藏起来&#xff0c;…

海南聚广众达电子商务咨询有限公司抖音开店怎么样?

在数字化浪潮汹涌的当下&#xff0c;电商行业正迎来前所未有的发展机遇。作为抖音电商服务领域的佼佼者&#xff0c;海南聚广众达电子商务咨询有限公司凭借其专业的服务团队和前瞻的战略眼光&#xff0c;成为了众多品牌和商家在抖音平台上实现业务增长的重要合作伙伴。今天&…

毫米波移动通信系统中的波束赋形—模数混合的波束赋形架构

模数混合的波束赋形将波束赋形分为了模拟波束赋形与数字波束赋形两部分&#xff0c;其模拟部分通过移相器实现&#xff0c;数字部分通过基带预编码实现&#xff0c;&#xff0c;其结构如图2所示。当射频链路数目NRF为1时&#xff0c;认为其是一种特殊的模数混合的波束赋形。 此…

linux企业级CDN/100万并发架构设计/企业故障案例、网站优化架构原则

高薪思想&#xff1a;财富来源于信息差 什么是cdn&#xff1f; cdn全称是contenct delivery network内容分发网络 cdn是一套分布式&#xff0c;缓存&#xff0c;集群&#xff0c;范围可以是全球或全国&#xff0c;运行的实质是通过智能DNS获取用户的来源地域以及上网线路 为…

perfect-scrollbar缩小浏览器窗口滚动条无线滚动的bug

https://github.com/mdbootstrap/perfect-scrollbar/issues/153