博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Python | Python学习之深浅拷贝
阅读量:6272 次
发布时间:2019-06-22

本文共 3273 字,大约阅读时间需要 10 分钟。

直接步入正题,聊一聊 Python 中的深浅拷贝

关于 is 和 ==

== 是 python 标准操作符中的比较操作符,用来比较判断两个对象的 value(值) 是否相等 。

# 例1.1a = '2332424'b = '2332424'print(a == b)输出:True[Finished in 0.1s]

is 也被叫做同一性运算符,这个运算符比较判断的是对象间的唯一身份标识,也就是id是否相同。

# 例1.2a = b = ['2343456']c = ['2343456']print(a == b, id(a), id(b))print(a == c, id(a), id(c))print(a is b, id(a), id(b))print(a is c, id(a), id(c))输出:True 10715336 10715336True 10715336 10717064True 10715336 10715336False 10715336 10717064[Finished in 0.1s]

不同类型下, == 的结果都为True,但是 is 的结果则不一定

# 例1.3# a 和 b 都为数值时a = 1b = 1print(id(a))print(id(b))print(a == b)print(a is b)17150776161715077616TrueTrue[Finished in 0.1s]# a 和 b 都为字符串时a = 'abcdefg'b = 'abcdefg'print(id(a))print(id(b))print(a == b)print(a is b)1184769611847696TrueTrue[Finished in 0.1s]# a 和 b 都为列表时a = ['12345678']b = ['12345678']print(id(a))print(id(b))print(a == b)print(a is b)1694125616942984TrueFalse[Finished in 0.1s]# a 和 b 都是字典时a = {'a':1,'b':2} b = {'a':1,'b':2} print(id(a))print(id(b))print(a == b)print(a is b)1042119210421256TrueFalse[Finished in 0.1s]# a 和 b 都是元组时a = (1,2,3)b = (1,2,3)print(id(a))print(id(b))print(a == b)print(a is b)1196060811960680TrueFalse[Finished in 0.1s]# a 和 b 都为集合时a = set([1,2,3])b = set([1,2,3])print(id(a))print(id(b))print(a == b)print(a is b)1685050416919464TrueFalse[Finished in 0.1s]

只有数值型和字符串型的情况下,a is b才为True,当a和b是tuple,list,dict或set型时,a is b为False。

关于深浅拷贝

首先我们来看几个栗子:

# 例2.1:赋值a = [1,2,3]b = aa.append([4,5,6])print(a)print(b)print(a is b)print(a == b)输出:[1, 2, 3, [4, 5, 6]][1, 2, 3, [4, 5, 6]][Finished in 0.1s]TrueTrue

在例2.1中a的末尾新增了[4,5,6]之后,b的值也发生了变化,这是因为a把值赋值给b只是将创建的b对象指向了a对象指向的内存,这时的a和b指向的都是同一块内存空间(id),所以修改a后b的值也一起改变。

# 例2.2:赋值a = [1,2,3]b = aa = [4,5,6]print(a)print(b)print(id(a))print(id(b))输出:[4, 5, 6][1, 2, 3]1084813610846408[Finished in 0.1s]

在例2.2中先将a赋值给b,又将[4,5,6]赋值给a,这时已经给a创建了新的内存空间,所以打印a,b时a和b的值并不相同。

下面来看下深浅拷贝的不同之处:

首先字符串 、元组以及数值的深浅拷贝是没有差别的,如下例子所示:

# 例3.1# 字符串import copya = '1345'b = copy.copy(a)c = copy.deepcopy(a)print(id(a), id(b), id(c))输出:17025040 17025040 17025040[Finished in 0.1s]# 数值import copya = 1345b = copy.copy(a)c = copy.deepcopy(a)print(id(a), id(b), id(c))输出:9326480 9326480 9326480[Finished in 0.1s]# 元组import copya = (1,2,3)b = copy.copy(a)c = copy.deepcopy(a)print(id(a), id(b), id(c))输出:10715424 10715424 10715424[Finished in 0.2s]

而对于字典、列表 进行浅拷贝和深拷贝时,其id是有变化的

# 列表import copya = [1,2,3]b = copy.copy(a)c = copy.deepcopy(a)print(id(a), id(b), id(c))输出:17540104 17040264 17419656[Finished in 0.1s]#字典import copya = {"a": "1", "b": 2, "c": ["c", 3]}b = copy.copy(a)c = copy.deepcopy(a)print(a, b, c)print(id(a), id(b), id(c))a['c'] = '666'print(a, b, c)print(id(a), id(b), id(c))输出:{'b': 2, 'a': '1', 'c': ['c', 3]} {'b': 2, 'a': '1', 'c': ['c', 3]} {'b': 2, 'a': '1', 'c': ['c', 3]}6882248 17105992 16886792{'b': 2, 'a': '1', 'c': '666'} {'b': 2, 'a': '1', 'c': ['c', 3]} {'b': 2, 'a': '1', 'c': ['c', 3]}6882248 17105992 16886792[Finished in 0.1s]

浅拷贝 :不管多么复杂的数据结构,浅拷贝都只会copy一层,且当copy指向不可变类型时,copy不会执行copy操作。当copy指向可变类型时,copy会执行第一层拷贝,即拷贝浅层的指向。

深拷贝 :其实深拷贝就是在内存中重新开辟一块空间,不管数据结构多么复杂,只要遇到可能发生改变的数据类型,就重新开辟一块内存空间把内容复制下来,直到最后一层,不再有复杂的数据类型,就保持其原引用。这样,不管数据结构多么的复杂,数据之间的修改都不会相互影响,这就是深拷贝,也同样可以将深拷贝理解为真正意义上的复制,是两个操作独立的个体。

有关深浅拷贝相关的图片说明可以参照下面几个链接的内容,以加深理解:

也可参照下面的几张图片:

image

image

image

原文发布时间为:2018-07-10

本文来自云栖社区合作伙伴“”,了解相关信息可以关注“”。

转载地址:http://fnvpa.baihongyu.com/

你可能感兴趣的文章
linux防火墙相关 iptables
查看>>
最简单的单例模式
查看>>
JPopupMenu的使用以及JPopupMenu中子组件的事件处理
查看>>
从反汇编的角度看引用和指针的区别
查看>>
拓马长枪定乾坤
查看>>
UIProgressView的详细使用
查看>>
Silverlight实用窍门系列:70.Silverlight的视觉状态组VisualStateGroup
查看>>
照片筛选与上传功能
查看>>
Hello ZED
查看>>
常见web攻击方式
查看>>
hdu 4472
查看>>
oracle存储过程中is和as区别
查看>>
windows 2003 群集
查看>>
几个gcc的扩展功能
查看>>
Spark一个简单案例
查看>>
关于结构体占用空间大小总结(#pragma pack的使用)
查看>>
通过浏览器查看nginx服务器状态配置方法
查看>>
shell简介
查看>>
android 使用WebView 支持播放优酷视频,土豆视频
查看>>
怎么用secureCRT连接Linux
查看>>