”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 浅拷贝与深拷贝——它们到底是什么? - 使用 JavaScript 和 Python 的示例

浅拷贝与深拷贝——它们到底是什么? - 使用 JavaScript 和 Python 的示例

发布于2024-08-31
浏览:835

介绍

在编程世界中,复制数据是一项常见任务。然而,并非所有副本都是一样的。经常出现的两个术语是浅拷贝深拷贝。了解它们之间的差异对于避免难以检测的错误至关重要。

什么是浅拷贝?

A 浅拷贝仅复制对象的第一层,留下对更深层次的原始数据的引用。这意味着如果原始对象内部有其他对象(嵌套),浅复制只会复制对这些对象的引用,而不是对象本身。

JavaScript 中的示例

const originalArray = [1, 2, [3, 4]];
const shallowCopy = originalArray.slice();

shallowCopy[2][0] = 99;

console.log(originalArray); // [1, 2, [99, 4]]
console.log(shallowCopy);   // [1, 2, [99, 4]]

Python 中的示例

import copy

original_list = [1, 2, [3, 4]]
shallow_copy = copy.copy(original_list)

shallow_copy[2][0] = 99

print(original_list)  # [1, 2, [99, 4]]
print(shallow_copy)   # [1, 2, [99, 4]]

提示:

当您知道不需要修改嵌套对象时,浅复制非常有用。与深层复制相比,它速度更快,消耗的内存更少。

笔记:

在 JavaScript 中,如果您使用 Array.slice() 或 Object.assign(),您就是在进行浅复制!

什么是深拷贝?

A 深层复制复制对象的所有级别,甚至复制嵌套结构。这意味着对副本所做的任何更改都不会影响原始对象。

JavaScript 中的示例

const originalArray = [1, 2, [3, 4]];
const deepCopy = JSON.parse(JSON.stringify(originalArray));

deepCopy[2][0] = 99;

console.log(originalArray); // [1, 2, [3, 4]]
console.log(deepCopy);      // [1, 2, [99, 4]]

Python 中的示例

import copy

original_list = [1, 2, [3, 4]]
deep_copy = copy.deepcopy(original_list)

deep_copy[2][0] = 99

print(original_list)  # [1, 2, [3, 4]]
print(deep_copy)      # [1, 2, [99, 4]]

提示:

如果您正在使用复杂或嵌套的数据结构,深度复制是避免不必要的副作用的最安全选择。

笔记:

在 Python 中,当您需要安全地复制复杂对象时,copy.deepcopy() 是您的朋友。

直接比较:浅复制与深复制

下面是浅拷贝和深拷贝的直接比较:

特征 浅复制 深复制
浅复制 是的
深拷贝 是的
对原始对象的修改会影响副本 是的
复杂 低的 高的

提示:

请记住,在处理复杂对象时,浅复制速度更快,但深复制更安全。

常见用例

何时使用浅复制

  • 当您使用对象或简单数据结构时。
  • 当你需要提高性能时,深度修改不是问题。
  • 示例:应用程序配置、临时数据镜像。

何时使用深层复制

  • 当您使用嵌套或复杂的数据结构时。
  • 当您需要确保对副本的更改不会影响原始版本时。
  • 示例:复杂的数据操作、需要高安全性和一致性的应用程序。

笔记:

浅拷贝非常适合复制轻量级应用程序设置或临时数据!

常见问题以及如何避免它们

浅拷贝问题

一个常见的错误是在数据嵌套时使用浅拷贝而不是深拷贝。这可能会导致对原始对象进行不必要的修改。

例子:

const originalArray = [1, 2, [3, 4]];
const shallowCopy = originalArray.slice();

shallowCopy[2][0] = 99;

console.log(originalArray); // [1, 2, [99, 4]] (¡No esperado!)

提示:

在决定浅复制还是深复制之前,请务必检查对象是否具有嵌套级别。

在 JavaScript 中进行复印的工具和函数

使用 Object.assign() 进行浅复制

const originalObject = { a: 1, b: { c: 2 } };
const shallowCopy = Object.assign({}, originalObject);

使用 ...spread 进行浅复制

const originalArray = [1, 2, 3];
const shallowCopy = [...originalArray];

使用 StructuredClone() 进行深度复制

const originalObject = { a: 1, b: { c: 2 } };
const deepCopy = structuredClone(originalObject);

提示:

structuralClone() 非常适合复制复杂或圆形结构,而不会伤脑筋。

使用 Lodash 等库进行深度复制

const _ = require('lodash');
const originalObject = { a: 1, b: { c: 2 } };
const deepCopy = _.cloneDeep(originalObject);

在 Python 中进行复印的工具和函数

使用复制模块

import copy

original_list = [1, 2, [3, 4]]
shallow_copy = copy.copy(original_list)
deep_copy = copy.deepcopy(original_list)

copy.copy() 和 copy.deepcopy() 之间的区别

  • copy.copy():浅复制。
  • copy.deepcopy():深层复制。

笔记:

在 Python 中,有时您只需要浅拷贝即可避免列表意外更改!

总结与结论

综上所述,浅拷贝和深拷贝都有各自的用途。关键是了解您正在使用的数据的结构并选择适当的复制方法。

常见问题解答

1. 浅拷贝总是比深拷贝快吗?

是的,因为它复制的数据较少。

2. 可以在没有外部库的情况下在 JavaScript 中进行深复制吗?

是的,使用 JSON.parse(JSON.stringify()) 或 StructuredClone()。

3. 如果我尝试修改嵌套在浅表副本中的对象,会发生什么情况?

原来的对象也会受到影响。

4. 是否总是使用深复制来避免问题更好?

不一定,仅当您使用复杂的数据结构时。

5. 与 JavaScript 中的其他深复制方法相比, StructuredClone() 有何优势?

它是原生的,支持循环结构并且比 JSON.parse(JSON.stringify()) 更高效,此外还允许将值完全从一个对象传输到另一个对象。


使用浅拷贝而不是深拷贝时的错误比你想象的更常见!我希望这个小指南可以帮助您避免复制数据时出现任何问题。

请在评论中告诉我,您是否已经了解深拷贝和浅拷贝以及您是否曾因它们而遇到过问题?


Shallow Copy vs Deep Copy - ¿Qué son realmente? - Ejemplos con JavaScript y Python

BYXN的笔记本? |子栈

我的公用笔记本! ???.单击以阅读 Substack 出版物 BYXN 的笔记本? 17天前推出。

Shallow Copy vs Deep Copy - ¿Qué son realmente? - Ejemplos con JavaScript y Python bhyxen.substack.com

照片由 Mohammad Rahmani 在 Unsplash 上拍摄

版本声明 本文转载于:https://dev.to/bhyxen/shallow-copy-vs-deep-copy-que-son-realmente-ejemplos-con-javascript-y-python-10ja?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-12-26
  • 在 Go 中使用 WebSocket 进行实时通信
    在 Go 中使用 WebSocket 进行实时通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSocke...
    编程 发布于2024-12-26
  • 除了“if”语句之外:还有什么地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有什么地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-12-26
  • 如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    在 PHP 中组合关联数组在 PHP 中,将两个关联数组组合成一个数组是一项常见任务。考虑以下请求:问题描述:提供的代码定义了两个关联数组,$array1和$array2。目标是创建一个新数组 $array3,它合并两个数组中的所有键值对。 此外,提供的数组具有唯一的 ID,而名称可能重合。要求是构...
    编程 发布于2024-12-26
  • 如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 识别今天生日的用户使用 MySQL 确定今天是否是用户的生日涉及查找生日匹配的所有行今天的日期。这可以通过一个简单的 MySQL 查询来实现,该查询将存储为 UNIX 时间戳的生日与今天的日期进行比较。以下 SQL 查询将获取今天有生日的所有用户: FROM USERS ...
    编程 发布于2024-12-26
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-12-26
  • HTML 格式标签
    HTML 格式标签
    HTML 格式化元素 **HTML Formatting is a process of formatting text for better look and feel. HTML provides us ability to format text without us...
    编程 发布于2024-12-26
  • 如何在 PHP 中转换所有类型的智能引号?
    如何在 PHP 中转换所有类型的智能引号?
    在 PHP 中转换所有类型的智能引号智能引号是用于代替常规直引号(' 和 ")的印刷标记。它们提供了更精致和然而,软件应用程序通常会在不同类型的智能引号之间进行转换,从而导致不一致。智能引号中的挑战转换转换智能引号的困难在于用于表示它们的各种编码和字符,不同的操作系统和软件程序采用...
    编程 发布于2024-12-26
  • 插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入记录时如何解决“一般错误:2006 MySQL 服务器已消失”介绍:将数据插入 MySQL 数据库有时会导致错误“一般错误:2006 MySQL 服务器已消失”。当与服务器的连接丢失时会出现此错误,通常是由于 MySQL 配置中的两个变量之一所致。解决方案:解决此错误的关键是调整wait_tim...
    编程 发布于2024-12-26
  • 循环 JavaScript 数组有哪些不同的方法?
    循环 JavaScript 数组有哪些不同的方法?
    使用 JavaScript 循环遍历数组遍历数组的元素是 JavaScript 中的一项常见任务。有多种方法可供选择,每种方法都有自己的优点和局限性。让我们探讨一下这些选项:数组1。 for-of 循​​环 (ES2015 )此循环使用迭代器迭代数组的值:const arr = ["a&q...
    编程 发布于2024-12-26
  • 尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    解决 PHP 中的 POST 请求故障在提供的代码片段中:action=''而不是:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"检查 $_POST数组:表单提交后使用 var_dump 检查 $_POST 数...
    编程 发布于2024-12-26
  • 如何在 Python 中有效地暂停 Selenium WebDriver 执行?
    如何在 Python 中有效地暂停 Selenium WebDriver 执行?
    Selenium WebDriver 中的等待和条件语句问题: 如何在 Python 中暂停 Selenium WebDriver 执行几毫秒?答案:虽然time.sleep() 函数可用于暂停执行指定的秒数,在 Selenium WebDriver 自动化中一般不建议使用。使用 Selenium ...
    编程 发布于2024-12-26
  • C++ 赋值运算符应该是虚拟的吗?
    C++ 赋值运算符应该是虚拟的吗?
    C 中的虚拟赋值运算符及其必要性 虽然赋值运算符可以在 C 中定义为虚拟,但这不是强制要求。然而,这种虚拟声明引发了关于虚拟性的必要性以及其他运算符是否也可以虚拟的问题。虚拟赋值运算符的案例赋值运算符本质上并不虚拟。然而,当将继承类的对象分配给基类变量时,它就变得必要了。这种动态绑定保证了调用基于对...
    编程 发布于2024-12-26
  • JavaScript 中的 Let 与 Var:范围和用法有什么区别?
    JavaScript 中的 Let 与 Var:范围和用法有什么区别?
    JavaScript 中的 Let 与 Var:揭秘范围和临时死区在 ECMAScript 6 中引入,let 语句引发了开发人员的困惑,特别是它与已建立的 var 关键字有何不同。本文深入研究了这两个变量声明之间的细微差别,重点介绍了它们的作用域规则和最佳用例。范围根本区别在于它们的作用域行为。用...
    编程 发布于2024-12-26
  • 如何使用 JavaScript 用逗号分割字符串,忽略双引号内的逗号?
    如何使用 JavaScript 用逗号分割字符串,忽略双引号内的逗号?
    使用 JavaScript 用逗号分割字符串,忽略双引号内的逗号解决用逗号分割字符串同时保留 double 的挑战- 引用段,我们可以在 JavaScript 中使用正则表达式。方法如下:var str = 'a, b, c, "d, e, f", g, h'; var arr ...
    编程 发布于2024-12-26

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3