Python 中浅拷贝和深拷贝的区别(python浅拷贝与深拷贝)
off999 2024-10-05 19:43 25 浏览 0 评论
引言
Python 附带了一个名为 copy 的模块,它提供了特定的复制功能。在本文中,我们将探索什么是深拷贝和浅拷贝。此外,我们还将讨论它们之间的差异以及何时使用其中一种而不是另一种。
不可变对象 vs 可变对象
在进入 Python 中的浅拷贝和深拷贝之前,首先要理解可变对象类型和不可变对象类型之间的区别。顾名思义,不可变对象是不可以被修改的,因此,当这些对象的值被修改时,Python 会创建一个新的对象。
例如,假设我们有两个变量引用同一个整数对象:
>>> a = 10
>>> b = a # variables a and b hold the reference to the same object
现在,如果我们对变量 a 执行任何类型的操作,并且考虑到 Python 中的整数是不可变的,那么结果将会创建一个保存新值的新对象。这意味着对象的旧值(以及引用它的所有变量)将保持不变:
>>> a = a + 1
>>> print(a)
11
>>> print(b)
10
另一方面,可变对象类型允许对对象值进行就地修改。这意味着,当修改可变对象类型的值时,保存对同一对象的引用的所有变量都会受到影响。例如,假设我们确实有以下列表
>>> list_1 = [1, 2, 3]
>>> list_2 = list_1
考虑到 Python 中的列表是可变的,如果我们改变这两个列表中的任何一个,这个操作也会对其他变量产生直接影响,因为它们都指向内存中相同的对象引用。
>>> list_1[0] = 0
>>> print(list_1)
[0, 2, 3]
>>> print(list_2)
[0, 2, 3]
常规赋值
复制对象最直接的方法是通过常规的赋值操作。假设我们有一下操作:
a = [1, 2, 3]
b = a
在这种情况下,变量 a 和 b 对同一个对象都有相同的引用。这意味着,如果这两个变量中的任何一个用于执行就地修改,其他变量也将受到影响。
>>> a[0] = 0
>>> print(a)
[0, 2, 3]
>>> print(b)
[0, 2, 3]
因此,当我们必须处理不可变的对象类型时,通常会使用常规的赋值操作。在这种情况下,当使用两个变量中的任何一个执行操作时,另一个变量将保持不变,因为它的引用指向的是不变的旧对象。
>>> id(a) == id(b)
True
Python 中的赋值语句不复制对象,它们在目标和对象之间创建绑定。
浅拷贝 vs 深拷贝
在深入讨论浅拷贝和深拷贝的细节之前,请注意,它们的区别只有在我们必须处理本质上是嵌套结构的复合对象时才有意义。换句话说,复合对象是包含其他对象的对象,例如,列表列表或集合字典。
一个浅拷贝将获得一个原始对象的副本并创建一个新的复合对象,但是如果我们正在复制的对象是一个复合对象,那么内部对象将与在原始对象中找到的对象相同。
>>> import copy
>>> b = copy.copy(a)
>>> id(a) == id(b)
False
如我们所见,列表对象 a 和 b 是不同的,这意味着它们持有指向内存中不同对象的不同引用(即使这些对象的值相同)。
当我们需要处理复合对象时,事情会变得有点复杂。现在让我们假设变量 a 是一个复合对象,它表示一个列表列表:
a = [[1, 2, 3], [4, 5, 6]]
现在让我们对 a 进行浅拷贝:
>>> import copy
>>> b = copy.copy(a)
我们可以看到 a 和 b 是不同的对象:
>>> id(a) == id(b)
False
然而,内部对象(即两个内部列表)与原始对象引用的对象相同:
>>> id(a[0]) == id(b[0])
True
这是非常危险的,因为任何内部列表的更改都会影响引用这些内部列表的其他复合对象:
>>> a[0][0] = 0
>>> a
[[0, 2, 3], [4, 5, 6]]
>>> b
[[0, 2, 3], [4, 5, 6]]
因此,只有当我们不必处理复合对象时,浅拷贝才适用。
浅拷贝构造一个新的复合对象,然后(在可能的范围内)将对原始对象中找到的对象的引用插入其中。
深层拷贝将获取原始对象的副本,然后递归地获取找到的内部对象的副本(如果有的话)。
>>> import copy
>>> a = [[1, 2, 3], [4, 5, 6]]
>>> b = copy.deepcopy(a)
同样,我们可以看到原始对象和复制对象在本质上是不同的:
>>> id(a) == id(b)
False
但在这种情况下,即使是内部对象也会不同:
>>> id(a[0]) == id(b[0])
False
这意味着 a 中任何嵌套列表的更改都不会影响对象 b 中的相应列表:
>>> a[0][0] = 0
>>> a
[[0, 2, 3], [4, 5, 6]]
>>> b
[[1, 2, 3], [4, 5, 6]]
因此,当我们必须处理复合对象并希望确保任何内部对象的更改都不会影响引用相同对象的其他变量时,深拷贝更为合适。
深拷贝构造一个新的复合对象,然后递归地将原始对象中找到的对象的副本插入其中。
总结
在本文中,我们探讨了用 Python 复制对象的三种基本方法。最初,我们讨论了不可变对象类型和可变对象类型之间的区别。不需要复制不可变物件类型,因为这些实例的值永远不会改变。另一方面,开发人员在修改可变对象类型时需要非常小心,因为这个操作可能会潜在地影响保存相同对象的引用的其他变量。当此类对象就地更改时,引用同一对象的所有其他变量也将受到此更改的影响。
因此,了解如何正确地复制可变对象以避免代码中的 bug 非常重要。回想一下,一个浅拷贝将从原始对象中创建一个新对象,但是如果对象包含其他对象,那么内部对象将不会被复制。另一方面,深度拷贝将为复合对象中包含的内部对象创建一个新对象。
相关推荐
- Python设计模式 第 13 章 中介者模式(Mediator Pattern)
-
在行为型模式中,中介者模式是解决“多对象间网状耦合”问题的核心模式。它就像“机场调度中心”——多个航班(对象)无需直接沟通起飞、降落时间,只需通过调度中心(中介者)协调,避免航班间的冲突与混乱...
- 1.3.1 python交互式模式的特点和用法
-
什么是Python交互模式Python交互模式,也叫Python交互式编程,是一种在Python解释器中运行的模式,它允许用户在解释器窗口中输入单个Python语句,并立即查看结果,而不需要编写整个程...
- Python设计模式 第 8 章 装饰器模式(Decorator Pattern)
-
在结构型模式中,装饰器模式是实现“动态功能扩展”的核心模式。它就像“手机壳与手机的关系”——手机(原始对象)具备通话、上网等基础功能,手机壳(装饰器)可在不改变手机本身的前提下,为其新增保护、...
- python设计模式 综合应用与实战指南
-
经过前面16章的学习,我们已系统掌握创建型模式(单例、工厂、建造者、原型)、结构型模式(适配器、桥接、组合、装饰器、外观、享元、代理)、行为型模式(责任链、命令、迭代器、中介者、观察者、状态、策略...
- Python入门学习教程:第 16 章 图形用户界面(GUI)编程
-
16.1什么是GUI编程?图形用户界面(GraphicalUserInterface,简称GUI)是指通过窗口、按钮、菜单、文本框等可视化元素与用户交互的界面。与命令行界面(CLI)相比,...
- Python 中 必须掌握的 20 个核心:str()
-
str()是Python中用于将对象转换为字符串表示的核心函数,它在字符串处理、输出格式化和对象序列化中扮演着关键角色。本文将全面解析str()函数的用法和特性。1.str()函数的基本用法1.1...
- Python偏函数实战:用functools.partial减少50%重复代码的技巧
-
你是不是经常遇到这样的场景:写代码时同一个函数调用了几十次,每次都要重复传递相同的参数?比如处理文件时总要用encoding='utf-8',调用API时固定传Content-Type...
- 第2节.变量和数据类型【第29课-输出总结】
-
同学们,关于输出的知识点讲解完成之后,把重点性的知识点做一个总结回顾。·首先对于输出这一章节讲解的比如有格式化符号,格式化符号这里需要同学们额外去多留意的是不是百分号s格式化输出字符串。当然课上也说百...
- AI最火语言python之json操作_python json.loads()
-
JSON(JavaScriptObjectNotation,JavaScript对象表示法)是一种开放标准的文件格式和数据交换格式,它易于人阅读和编写。JSON是一种常用的数据格式,比如对接各种第...
- python中必须掌握的20个核心函数—split()详解
-
split()是Python字符串对象的方法,用于将字符串按照指定的分隔符拆分成列表。它是文本处理中最常用的函数之一。一、split()的基本用法1.1基本语法str.split(sep=None,...
- 实用方法分享:pdf文件分割方法 横向A3分割成纵向A4
-
今天在街上打印店给儿子打印试卷时,我在想:能不能,把它分割成A4在家中打印,这样就不需要跑到街上的打印店打印卷子了。原来,老师发的作业,是电子稿,pdf文件,A3格式的试卷。可是家中的打印机只能打印A...
- 20道常考Python面试题大总结_20道常考python面试题大总结免费
-
20道常考Python面试题大总结关于Python的面试经验一般来说,面试官会根据求职者在简历中填写的技术及相关细节来出面试题。一位拿了大厂技术岗SpecialOffer的网友分享了他总结的面试经...
- Kotlin Data Classes 快速上手_kotlin快速入门
-
引言在日常开发中,我们常常需要创建一些只用来保存数据的类。问题是,这样的类往往需要写一堆模板化的方法:equals()、hashCode()、toString()……每次都重复,既枯燥又容易出错。//...
- python自动化RobotFramework中Collections字典关键字使用(五)
-
前言介绍安装好robotframework库后,跟之前文章介绍的BuiltIn库一样BuiltIn库使用介绍,在“python安装目录\Lib\site-packages\robot\librarie...
- Python中numpy数据分析库知识点总结
-
Python中numpy数据分析库知识点总结二、对已读取数据的处理②指定一个值,并对该值双边进行修改③指定两个值,并对第一个值的左侧和第二个值的右侧进行修改2.4数组的拼接和行列交换①竖直拼接(np...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- Python设计模式 第 13 章 中介者模式(Mediator Pattern)
- 1.3.1 python交互式模式的特点和用法
- Python设计模式 第 8 章 装饰器模式(Decorator Pattern)
- python设计模式 综合应用与实战指南
- Python入门学习教程:第 16 章 图形用户界面(GUI)编程
- Python 中 必须掌握的 20 个核心:str()
- Python偏函数实战:用functools.partial减少50%重复代码的技巧
- 第2节.变量和数据类型【第29课-输出总结】
- AI最火语言python之json操作_python json.loads()
- python中必须掌握的20个核心函数—split()详解
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)