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

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

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

介绍

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

什么是浅拷贝?

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刪除
最新教學 更多>
  • React 中的 UseEffect
    React 中的 UseEffect
    歡迎 React Hooks 的世界!今天,我們將深入探討最受歡迎的掛鉤之一:useEffect。別擔心,我們會讓它變得有趣且易於理解。那麼,就讓我們開始吧! ? ?什麼是useEffect useEffect 是一個 React Hook,可讓您在功能元件中執行副作用。副作用是在元件外部發生的操...
    程式設計 發佈於2024-11-06
  • 如何在 Google Cloud Platform 免費層上建立現代資料平台
    如何在 Google Cloud Platform 免費層上建立現代資料平台
    我在 Medium.com 上發布了一系列七篇免費公開文章「如何在 Google Cloud Platform 免費層上建立現代資料平台」。 主要文章位於:https://medium.com/@markwkiehl/building-a-data-platform-on-gcp-0427500f...
    程式設計 發佈於2024-11-06
  • 貼文 #f 掙扎
    貼文 #f 掙扎
    這篇文章是關於我迄今為止在編碼和學習方面的掙扎 一個。我只能保持專註一個小時,最多兩個小時。 b.我很容易分心 c.我不能久坐,否則我會開始感到煩躁和休息腿部問題。 我想到的有助於解決問題的解決方案 一個。我需要開始更頻繁地使用我的番茄工作法應用程式 B. 我開始將手機調成震動,如果我有另一個螢...
    程式設計 發佈於2024-11-06
  • 面向 Web 開發人員的熱門 Chrome 擴充功能 4
    面向 Web 開發人員的熱門 Chrome 擴充功能 4
    2024 年最適合 Web 開發者的 10 款 Chrome 擴展 隨著 2024 年的進展,Chrome 擴充功能已成為 Web 開發人員工具包中不可或缺的一部分,在瀏覽器中提供強大的功能。在這篇文章中,我們將探討今年在 Web 開發社群掀起波瀾的 10 大 Chrome 擴充功...
    程式設計 發佈於2024-11-06
  • 如何使用 React Router v4/v5 巢狀路由:簡化指南
    如何使用 React Router v4/v5 巢狀路由:簡化指南
    React Router v4/v5 的嵌套路由:簡化指南使用React Router 時,嵌套路由是組織的關鍵技術您的應用程式的導航。然而,新手經常面臨設定嵌套路由的挑戰。本文旨在簡化使用 React Router v4/v5 的流程。 React Router v4 在路由嵌套方式上引入了重大轉...
    程式設計 發佈於2024-11-06
  • 如何使用 UTF8 字元編碼保留 MySQL 中的表格式?
    如何使用 UTF8 字元編碼保留 MySQL 中的表格式?
    使用UTF8 字元編碼增強MySQL 命令列格式使用儲存在資料庫表中的瑞典語和挪威語字串時,查詢資料時可能會遇到表格式問題使用不同的字元集。 問題陳述預設情況下,使用「set names latin1;」產生失真的輸出: ----------------------------------- | ...
    程式設計 發佈於2024-11-06
  • CSS 盒子模型
    CSS 盒子模型
    CSS 盒子模型是 Web 開發中的一個基本概念,它構成了 Web 佈局和設計的基礎。它決定了元素的大小、內容的呈現方式以及它們如何在網頁上相互互動。掌握盒模型對於任何使用 HTML 和 CSS 的開發人員來說都是至關重要的,因為它會影響元素的顯示、間隔和對齊方式。 在本文中,我們將詳細探討 CS...
    程式設計 發佈於2024-11-06
  • 我如何寫 CSS 選擇器
    我如何寫 CSS 選擇器
    有很多 CSS 方法,但我討厭它們。有些多(順風等),有些少(BEM、OOCSS 等)。但歸根結底,它們都有缺陷。 當然,人們使用這些方法有充分的理由,並且解決的許多問題我也遇到過。因此,在這篇文章中,我想寫下我自己的關於如何保持 CSS 組織的指南。 這不是任何人都可以開始使用的完整描述的 C...
    程式設計 發佈於2024-11-06
  • 為什麼輸入元素不支援 HTML5 中的 ::after 偽元素?
    為什麼輸入元素不支援 HTML5 中的 ::after 偽元素?
    ::before 和::after 的偽元素相容性在HTML5 中,::before 和::after 偽元素可以使用附加內容(例如圖示或複選標記)增強元素。然而,並非所有元素都完全支援這些偽元素。 輸入元素和 ::after在提供的範例中,::after 偽元素不是顯示在輸入元素上。這是因為類似 ...
    程式設計 發佈於2024-11-06
  • 如何使用 PHP 決定特定時區的星期?
    如何使用 PHP 決定特定時區的星期?
    在PHP 中確定指定時區的星期幾在PHP 中處理日期和時間時,處理起來可能具有挑戰性時區並計算具體值,例如星期幾。本文將引導您完成使用 PHP 尋找特定時區中的星期幾的過程。 了解使用者的時區要確定使用者的時區,您需要使用 PHP 函數 date_default_timezone_get()。此函數...
    程式設計 發佈於2024-11-06
  • 如何在 Go 頻道中有效地產生不同的值?
    如何在 Go 頻道中有效地產生不同的值?
    在 Go Channel 中高效產生不同值在 Go 中,Channel 為並發通訊提供了強大的機制。但是,在使用通道時,您可能會遇到需要過濾掉重複值或確保僅發出不同值的情況。本文探討了建立僅輸出唯一值的通道的有效方法。 產生不同值的挑戰考慮以下場景:您有一個通道接收多個值,並且您希望迭代它,同時僅打...
    程式設計 發佈於2024-11-06
  • 如何使用 Tailwind CSS 設定 os Next.js
    如何使用 Tailwind CSS 設定 os Next.js
    若要使用 Tailwind CSS 設定 Next.js,請依照下列步驟: 第 1 步:建立一個新的 Next.js 項目 如果您尚未建立 Next.js 項目,您可以使用 create-next-app 建立專案。 npx create-next-app@latest my-ne...
    程式設計 發佈於2024-11-06
  • 如何解決 PHPmailer HTML 內容渲染問題?
    如何解決 PHPmailer HTML 內容渲染問題?
    PHPmailer 無法渲染HTML 內容使用PHPmailer 發送電子郵件時,使用者遇到HTML 程式碼顯示為原始文字的問題交貨時。儘管使用了 IsHTML() 方法,所需的 HTML 內容仍然無法存取。 潛在問題此行為背後的原因在於方法呼叫的順序。與它的前身不同,PHPMailer 6 要求在...
    程式設計 發佈於2024-11-06
  • 如何使用 Java 從 HTML 文件中提取資料?
    如何使用 Java 從 HTML 文件中提取資料?
    Java HTML解析要從網站取得數據,首先必須了解HTML文件的結構。 HTML 元素使用標籤進行組織,標籤指定每個元素的類型和內容。 例如,以下HTML 表示具有特定CSS 類別的div 標籤:<div class="classname"></div>...
    程式設計 發佈於2024-11-06
  • 為什麼 Java 異常處理程式碼會產生「132Exception in thread main MyExc1」而不是「13Exception in thread main MyExc2」?
    為什麼 Java 異常處理程式碼會產生「132Exception in thread main MyExc1」而不是「13Exception in thread main MyExc2」?
    Java中的異常處理:解開歧義Java中的異常處理:解開歧義// Exception Heirarchy class MyExc1 extends Exception {} class MyExc2 extends Exception {} class MyExc3 extends MyExc2 {...
    程式設計 發佈於2024-11-06

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

Copyright© 2022 湘ICP备2022001581号-3