」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 淺拷貝與深拷貝──它們到底是什麼? - 使用 JavaScript 和 Python 的範例

淺拷貝與深拷貝──它們到底是什麼? - 使用 JavaScript 和 Python 的範例

發佈於2024-08-31
瀏覽:729

介绍

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

什么是浅拷贝?

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如有侵犯,請聯絡study_golang @163.com刪除
最新教學 更多>
  • 插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入記錄時如何解決「一般錯誤:2006 MySQL 伺服器已消失」介紹:將資料插入MySQL 資料庫有時會導致錯誤「一般錯誤:2006 MySQL 伺服器已消失」。當與伺服器的連線遺失時會出現此錯誤,通常是由於 MySQL 配置中的兩個變數之一所致。 解決方案:解決此錯誤的關鍵是調整wait_tim...
    程式設計 發佈於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
  • 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
  • 如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    在 PHP 中組合關聯數組在 PHP 中,將兩個關聯數組組合成一個數組是常見任務。考慮以下請求:問題描述:提供的代碼定義了兩個關聯數組,$array1和$array2。目標是建立一個新陣列 $array3,它合併兩個陣列中的所有鍵值對。 此外,提供的陣列具有唯一的 ID,而名稱可能重疊。要求是建構一...
    程式設計 發佈於2024-12-26
  • 大批
    大批
    方法是可以在物件上呼叫的 fns 數組是對象,因此它們在 JS 中也有方法。 slice(begin):將陣列的一部分提取到新數組中,而不改變原始數組。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index ...
    程式設計 發佈於2024-12-26
  • 在 Go 中使用 WebSocket 進行即時通信
    在 Go 中使用 WebSocket 進行即時通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSocke...
    程式設計 發佈於2024-12-26
  • 如何在 PHP 中轉換所有類型的智慧引號?
    如何在 PHP 中轉換所有類型的智慧引號?
    在 PHP 中轉換所有類型的智慧引號智慧引號是用來取代常規直引號(' 和")的印刷標記。它們提供了更精緻和然而,軟體應用程式通常會在不同類型的智能引號之間進行轉換,從而導致不一致。智能引號中的挑戰轉換轉換智慧引號的困難在於用於表示它們的各種編碼和字符,不同的作業系統和軟體程式採用自...
    程式設計 發佈於2024-12-26
  • 循環 JavaScript 陣列有哪些不同的方法?
    循環 JavaScript 陣列有哪些不同的方法?
    使用 JavaScript 迴圈遍歷陣列遍歷陣列的元素是 JavaScript 中常見的任務。有多種方法可供選擇,每種方法都有自己的優點和限制。讓我們探討一下這些選項:陣列1。 for-of 遵循(ES2015 )此循環使用迭代器迭代數組的值:const arr = ["a", ...
    程式設計 發佈於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 自動化中一般不建議使用。 使用 Seleniu...
    程式設計 發佈於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 = str....
    程式設計 發佈於2024-12-26
  • JavaScript 函數表達式中的感嘆號 (!) 有何作用?
    JavaScript 函數表達式中的感嘆號 (!) 有何作用?
    揭示函數表達式中感嘆號的用途在JavaScript 中,執行程式碼時,前面遇到感嘆號(!)函數可能會引發一些問題。讓我們深入研究一下它的功能及其在語法中的作用。 JavaScript 的語法規定,以「function foo() {}」形式宣告的函數是函數聲明,需要呼叫才能執行。然而,預處理帶有感嘆...
    程式設計 發佈於2024-12-26
  • 如何在 Go 中以程式設計方式存取文件組 ID (GID)?
    如何在 Go 中以程式設計方式存取文件組 ID (GID)?
    在Go 中訪問文件組ID (GID)在Go 中,os.Stat() 函數檢索文件信息,包括其系統資訊-特定屬性。此資訊儲存在 syscall.Sys 介面中。雖然列印介面直接顯示 GID,但以程式設計方式存取它會帶來挑戰。 要以 Linux 系統的字串形式取得 GID:file_info, _ :=...
    程式設計 發佈於2024-12-26

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3