拷贝的艺术:深拷贝与浅拷贝的区别与应用(上)

news2025/1/11 8:18:03

在这里插入图片描述

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6
🍨 阿珊和她的猫_CSDN个人主页
🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》
🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入门到实战全面掌握 uni-app》

文章目录

  • 一、引言
    • 介绍深拷贝和浅拷贝的背景
    • 解释为什么需要了解这两种拷贝方式
  • 二、深拷贝的基本概念
    • 解释什么是深拷贝
    • 深拷贝的实现原理
    • 深拷贝的应用场景
  • 三、浅拷贝的基本概念
    • 解释什么是浅拷贝
    • 浅拷贝的实现原理
    • 浅拷贝的应用场景

一、引言

介绍深拷贝和浅拷贝的背景

深拷贝和浅拷贝是在编程中常见的概念,它们用于处理对象的复制。深拷贝和浅拷贝的背景如下:

  • 深拷贝相当于创建了一个新的对象,这个对象的所有内容都和被拷贝的对象一模一样,即两者的修改是隔离的,相互之间没有影响。
  • 浅拷贝也是创建了一个对象,但是这个对象的某些内容(比如A)依然是被拷贝对象的,即通过这两个对象中任意一个修改A,两个对象的A都会受到影响。

在实际应用中,开发者需要根据具体的需求选择适当的拷贝方式,以确保程序的正确性和性能。

解释为什么需要了解这两种拷贝方式

了解深拷贝和浅拷贝的原因如下:

  1. 数据独立性:深拷贝可以确保复制后的对象与原始对象完全独立。当修改其中一个对象时,不会影响到另一个对象。这在处理复杂的数据结构或共享资源时非常重要,避免了意外的副作用和数据不一致。

  2. 性能考虑:浅拷贝通常比深拷贝更高效,因为它只复制了对象的引用,而不是复制对象的所有内容。在一些情况下,如对象包含大量数据或资源时,深拷贝可能会导致性能下降。

  3. 避免循环引用:在一些复杂的数据结构中,对象之间可能存在循环引用。如果使用浅拷贝,可能会导致无限循环或内存泄漏。深拷贝可以避免这种情况的发生,确保对象的复制是安全的。

总之,了解深拷贝和浅拷贝的概念和区别对于编写可靠、高效的代码非常重要。选择适当的拷贝方式可以根据具体的需求和场景来决定,以确保数据的正确性、独立性和性能。

二、深拷贝的基本概念

解释什么是深拷贝

深拷贝是一种拷贝对象的方式,它会创建一个完全独立于原始对象的新对象。深拷贝不仅复制了对象的引用,还复制了对象的所有内部属性和子对象。

在深拷贝过程中,新对象的属性和子对象与原始对象的属性和子对象是完全分离的,它们之间没有任何关联。修改新对象的属性或子对象不会影响原始对象,反之亦然。

深拷贝通常用于确保对象的独立性和数据的安全性。当需要保留原始对象的完整状态,并且在修改新对象时不会对原始对象造成影响时,深拷贝是非常有用的。

要实现深拷贝,通常需要递归地复制对象的所有属性和子对象。这可以通过手动编写拷贝函数或使用一些库(如lodashjQuery)提供的深拷贝方法来完成。

下面是一个使用 JavaScript 实现深拷贝的示例代码:

function deepCopy(source) {
  // 创建一个空对象作为目标对象
  let target = {};

  // 遍历源对象的属性
  for (let key in source) {
    if (source.hasOwnProperty(key)) {
      // 如果源对象的属性是对象,则进行递归深拷贝
      if (typeof source[key] === 'object' && source[key] !== null) {
        target[key] = deepCopy(source[key]);
      } else {
        // 否则,直接复制属性的值
        target[key] = source[key];
      }
    }
  }

  return target;
}

// 示例用法
const obj1 = { prop1: 'value1', prop2: { subProp: 'subValue' } };
const copiedObj = deepCopy(obj1);

console.log(obj1 === copiedObj);  // false
console.log(obj1.prop2 === copiedObj.prop2);  // false

在上述示例中,deepCopy函数接受一个源对象作为参数,并返回一个深拷贝后的新对象。它通过递归遍历源对象的属性,并对每个对象属性进行深拷贝,确保创建的新对象与源对象完全独立。

需要注意的是,深拷贝可能会消耗更多的内存和时间,并且对于一些特殊类型(如函数、循环引用等)的对象可能无法进行深拷贝。在实际应用中,需要根据具体情况选择是否使用深拷贝。

深拷贝的实现原理

深拷贝的实现原理是在进行拷贝时,不仅复制对象本身,还会递归地复制对象内部所有的引用对象。即深度克隆了所有引用类型的属性,创建了一个完全独立的新对象,该对象与原始对象没有任何关联,对新对象和原始对象的修改互不影响。

在深拷贝中,新对象和原始对象分别对应不同的内存区域,它们之间不存在引用关系,因此修改其中一个对象不会影响到另一个对象。实现深拷贝有多种方式,如通过实现Cloneable接口和重写clone()方法、通过序列化和反序列化对象等。

深拷贝的应用场景

深拷贝主要应用于以下场景:

  1. 避免对象共享:在一些情况下,需要确保两个对象完全独立,不共享任何内部状态。深拷贝可以创建对象的副本,使得修改副本不会影响原始对象。

  2. 数据保护:当需要对一个对象进行修改时,为了避免意外修改原始对象,可以使用深拷贝创建一个副本进行操作。

  3. 缓存和持久化:深拷贝可以用于将对象缓存到其他地方,或者将对象序列化到磁盘或数据库中进行持久化。

  4. 传递对象:在函数参数传递或在不同模块之间传递对象时,使用深拷贝可以确保传递的是对象的独立副本,而不是原始对象的引用。

  5. 多线程编程:在多线程环境中,为了避免竞态条件和数据不一致性,有时需要使用深拷贝来创建线程安全的对象副本。

在这里插入图片描述

需要注意的是,深拷贝会创建对象的完整副本,包括所有嵌套对象和引用对象。这可能会导致性能消耗和内存开销的增加,因此在使用深拷贝时需要考虑性能和资源的影响,并根据实际需求进行选择。

三、浅拷贝的基本概念

解释什么是浅拷贝

浅拷贝是一种拷贝对象的方式,它只复制了对象的第一层属性,而不复制嵌套对象或引用对象。

在浅拷贝过程中,新对象的属性将指向原始对象对应属性的引用,而不是创建新的嵌套对象或引用对象。这意味着,如果修改新对象的属性,原始对象的相应属性也会受到影响,因为它们共享同一个嵌套对象或引用对象。

以下是一个使用 JavaScript 实现浅拷贝的示例代码:

function shallowCopy(source) {
  let target = {};
  for (let key in source) {
    if (source.hasOwnProperty(key)) {
      target[key] = source[key];
    }
  }
  return target;
}

// 示例用法
const obj1 = { prop1: 'value1', prop2: { subProp: 'subValue' } };
const copiedObj = shallowCopy(obj1);

console.log(obj1 === copiedObj);  // false
console.log(obj1.prop2 === copiedObj.prop2);  // true

在上述示例中,shallowCopy函数创建了一个新的对象target,并将source对象的属性逐个复制到target中。然而,由于使用的是浅拷贝,嵌套对象prop2并没有被复制,而是共享了原始对象的引用。

浅拷贝通常比深拷贝更高效,因为它不需要递归复制嵌套对象。但在处理嵌套对象时需要特别小心,因为修改新对象的嵌套属性可能会影响原始对象。如果需要完全独立的副本,通常使用深拷贝更合适。

浅拷贝的实现原理

浅拷贝的实现原理是创建一个新的对象,然后将原始对象的属性逐个复制到新对象中。然而,对于嵌套对象或引用对象,浅拷贝只会复制引用,而不是创建新的嵌套对象或引用对象。

以下是一个使用 JavaScript 实现浅拷贝的示例代码:

function shallowCopy(source) {
  let target = {};
  for (let key in source) {
    if (source.hasOwnProperty(key)) {
      target[key] = source[key];
    }
  }
  return target;
}

// 示例用法
const obj1 = { prop1: 'value1', prop2: { subProp: 'subValue' } };
const copiedObj = shallowCopy(obj1);

console.log(obj1 === copiedObj);  // false
console.log(obj1.prop2 === copiedObj.prop2);  // true

在上述示例中,shallowCopy函数创建了一个新的对象target,并将source对象的属性逐个复制到target中。然而,由于使用的是浅拷贝,嵌套对象prop2并没有被复制,而是共享了原始对象的引用。

浅拷贝通常比深拷贝更高效,因为它不需要递归复制嵌套对象。但在处理嵌套对象时需要特别小心,因为修改新对象的嵌套属性可能会影响原始对象。如果需要完全独立的副本,通常使用深拷贝更合适。

浅拷贝的应用场景

浅拷贝主要适用于以下场景:

  1. 数据备份:在需要对数据进行备份时,可以使用浅拷贝来创建原始数据的副本,以防止意外修改原始数据。

  2. 数据传递:当需要将数据传递给其他函数或模块时,可以使用浅拷贝来避免共享嵌套对象或引用对象导致的数据修改。

  3. 缓存和暂存:浅拷贝可以用于创建对象的缓存或暂存副本,以减少重复计算或资源消耗。

在这里插入图片描述

需要注意的是,浅拷贝适用于不需要修改嵌套对象或引用对象的情况。如果需要对嵌套对象进行独立修改,通常使用深拷贝更合适。在实际应用中,根据具体需求选择适当的拷贝方式。

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

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

相关文章

轻量封装WebGPU渲染系统示例<50>- Json数据描述材质等场景信息

当前示例源码github地址: https://github.com/vilyLei/voxwebgpu/blob/feature/material/src/voxgpu/sample/DataDrivenScene2.ts 当前示例运行效果: 此示例基于此渲染系统实现,当前示例TypeScript源码如下: json场景数据: {"renderer": {"mtplE…

设计模式——组合模式(结构型)

引言 组合模式是一种结构型设计模式, 你可以使用它将对象组合成树状结构, 并且能像使用独立对象一样使用它们。 问题 如果应用的核心模型能用树状结构表示, 在应用中使用组合模式才有价值。 例如, 你有两类对象: ​…

搞懂这6 个持续集成工具,领先80%测试人

📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…

太空旅行:计算机技术的崭新航程

太空旅行:计算机技术的崭新航程 一、引言 自古以来,人类就对浩渺的宇宙充满了无尽的好奇和渴望。随着科技的飞速发展,太空旅行已经从科幻小说中的构想变为现实。在这个过程中,计算机技术起到了不可或缺的作用。从阿波罗时代的初…

12V升18V4A同步升压恒压WT3210

12V升18V4A同步升压恒压WT3210 WT3210 是一款高功率密度的全集成同步升压转换器,内部集成的功率MOSFET管导通电阻为上管8mΩ和下管15mΩ。可为便携式系统提供空间小尺寸 解决方案。WT3210具有 2.7V 至 20V 的宽输入电压范围,应用在单节或两节锂电池的便携…

数据可视化?这些平台能处

图表在各行各业都起到举重若轻的作用,无论是项目汇报、业绩分析,亦或是数据挖掘、统计分析,良好的可视化可以为我们的阐述起到画龙点睛的效果。在一篇文章中,如果只有密密麻麻的文字堆积,无论是谁恐怕都无法长期保持注…

[Verilog] Verilog 简介

主页: 元存储的博客 全文 2100 字。 文章目录 前言1. Verilog 简介2. 主要特性3. 应用领域总结 前言 Verilog HDL(简称 Verilog )是一种硬件描述语言,用于数字电路的系统设计。可对算法级、门级、开关级等多种抽象设计层次进行建…

GZ015 机器人系统集成应用技术样题3-学生赛

2023年全国职业院校技能大赛 高职组“机器人系统集成应用技术”赛项 竞赛任务书(学生赛) 样题3 选手须知: 本任务书共 26页,如出现任务书缺页、字迹不清等问题,请及时向裁判示意,并进行任务书的更换。参赛队…

seaborn库图形进行数据分析(基于tips数据集)

Seaborn 是一个基于 matplotlib 的数据可视化库,可以用来绘制各种统计图表,包括散点图、条形图、折线图、箱线图等。Seaborn 提供了一些用于美化图表的默认样式和颜色主题,使得生成的图表更具有吸引力。下面是一些 Seaborn 库的常用功能和用法…

性能测试基础知识总结

1、CPU 使用率 除了空闲时间外的其他时间占总 CPU 时间的百分比,就是CPU 使用率,即 1- 空闲时间/CPU 总时间。 当计算 CPU 使用率时,我们通常使用 /proc/stat 文件中的数据。该文件提供了有关 CPU 的计数器信息,包括各种状态下的…

HPM6750系列--第九篇 GPIO详解(基本操作)

一、目的 在之前的博文中我们主要介绍了不同系统不同开发编译调试环境的配置和操作(命令行方式、Visual Studio Code、Segger Embedded Studio for RISC-V),以帮助大家准备好学习环境为目的,但是未涉及到芯片本身以及外设的讲解。…

CCF编程能力等级认证GESP—C++2级—20230318

CCF编程能力等级认证GESP—C2级—20230318 单选题(每题 2 分,共 30 分)判断题(每题 2 分,共 20 分)编程题 (每题 25 分,共 50 分)画三角形百鸡问题 答案及解析单选题判断题编程题1编程题2 单选题…

【MySQL学习之基础篇】概述

文章目录 1. mysql的启动和停止命令2. 客户端连接3. 数据模型 1. mysql的启动和停止命令 通过指令启动或停止,以管理员身份运行cmd,进入命令行执行如下指令: (1)启动myaql net start mysql(2)…

解决App Store上架提示您必须上传 12.9 英寸 iPad Pro(第 2 代)显示屏的截屏

出错场景 在App Store Connect中,上架App时,出现以下错误提示. 要开始审核流程,必须提供以下项目:您必须上传 12.9 英寸 iPad Pro(第 2 代)显示屏的截屏。(2048,2732)您…

netty-daxin-2(netty常用事件讲解)

文章目录 netty常用事件讲解ChannelHandler接口ChannelHandler适配器类ChannelInboundHandler 子接口Channel 的状态调用时机ChannelHandler 生命周期示例NettServer&CustomizeInboundHandlerNettyClient测试分析 ChannelInboundHandlerAdapter适配器类SimpleChannelInboun…

【MySQL学习之基础篇】多表查询

文章目录 1. 多表关系1.1. 一对多1.2. 多对多1.3. 一对一 2. 多表查询概述2.1. 数据准备2.2. 概述 3. 查询的分类3.1. 内连接查询3.2. 外连接查询3.3. 自连接3.3.1. 自连接查询3.3.2. 联合查询 3.4. 子查询3.4.1. 概述3.4.2. 标量子查询3.4.3. 列子查询3.4.4. 行子查询3.4.5. 表…

等等Domino 14.0FP1

大家好,才是真的好。 节奏确实太快了,有时候我深感我也追不上。 以前Notes Domino是三年磨一剑,也就说每三年才发一个大版本,从2019年开始,进行了高频提速,居然一年一个大版本! 周末&#xf…

vue-json-excel 在导出一行a-table明细后再去导出另一行明细但导出结果还是和第一次一样

通过接口等拿到数据后要使用setTimeout稍微延时下再去触发导出,不然数还未更新就导出了

分库分表以后,如何实现扩容?

在实际开发中,数据库的扩容和不同的分库分表规则直接相关,今天我们从系统设计的角度,抽象了一个项目开发中出现的业务场景,从数据库设计、路由规则,以及数据迁移方案的角度进行讨论。 从业务场景出发进行讨论 假设这…

LeetCode(60)K 个一组翻转链表【链表】【困难】

目录 1.题目2.答案3.提交结果截图 链接: K 个一组翻转链表 1.题目 给你链表的头节点 head ,每 k 个节点一组进行翻转,请你返回修改后的链表。 k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍&#xf…