设为首页 收藏本站
查看: 891|回复: 0

[经验分享] 深入Python(4):深拷贝和浅拷贝

[复制链接]
发表于 2015-4-25 11:52:56 | 显示全部楼层 |阅读模式
一、前奏:熟悉Python内存管理
  在Python中,变量在第一次赋值时自动声明,在创建---也就是赋值的时候,解释器会根据语法和右侧的操作数来决定新对象的类型。
  引用计数器:一个内部跟踪变量
  引用计数:每一个对象各有多少个引用
  当对象被创建并(将其引用)赋值给变量时,该对象的引用计数就被设置为 1



>>> x = 3.14
  语句 x=3.14,创建一个浮点型对象并将其引用赋值给了x,x是第一个引用,该对象的引用计数为1
  当一个对象(的引用)又被赋值到其他变量,或做参数传递等,该对象的一个新的引用(或叫别名)被创建,则该对象的引用计数自动+1。
  以下都会增加引用计数:



y = x   #做别名
foo(x)  #做参数传递
mylis = [1,2,x,'a'] #成为容器对象的一个元素
  
  以下都会减少引用计数:



del x   #del显式销毁

bar = x
x = True    #对象的一个别名被赋值给其他对象

mylis.remove(x) #对象被从窗口对象中移除
del mylis   #窗口对象本身被销毁
  
二、Python的复制
  从上面可见,对象的赋值实际上是对象的引用。当创建一个对象,然后把它赋给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用。
  当你对一个对象赋值的时候(做为参数传递,或者做为返回值),Python和Java一样,总是传递原始对象的引用,而不是一个副本。



"""传递原始对象的引用,而不是一个副本"""
a = [1,2,3]
b = a
b.append(100)
print b         #[1, 2, 3, 100]
print a         #[1, 2, 3, 100]
print id(a)     #11530368
print id(b)     #11530368
  如 果你想修改一个对象,而且想让原始的对象不受影响,那你就需要对象复制。
  可以 使用copy.copy(),它可以进行对象的浅复制(shallow copy),它复制了对象,但对于对象中的元素,依然使用引用.
  (1)、使用切片[:]操作进行拷贝
  (2)、使用工厂函数(如list/dir/set)等进行拷贝
  (3)、copy.copy()



>>> jack = ['jack',['age',20]]
>>> tom = jack[:]
>>> anny = list(jack)
>>> jack
['jack', ['age', 20]]
>>> tom
['jack', ['age', 20]]
>>> anny
['jack', ['age', 20]]
>>> print id(jack),id(tom),id(anny)
13457088 18487376 18489136
  接下来修改上面例子,对姓名和年级进行修改:



>>> tom[0]='tom'
>>> anny[0]='anny'
>>> print tom
['tom', ['age', 20]]
>>> print anny
['anny', ['age', 20]]
>>> anny[1][1]
20
>>> anny[1][1]= 18
>>> anny[1][1]
18
>>> print jack,tom,anny
['jack', ['age', 18]] ['tom', ['age', 18]] ['anny', ['age', 18]]
  发现,虽然姓名都对号了,但是年龄却都变成了18.这是为什么呢?
  我们看看它们元素的id



>>> [id(x) for x in jack]
[13463040, 13456608]
>>> [id(x) for x in tom]
[13463424, 13456608]
>>> [id(x) for x in anny]
[18501664, 13456608]
  发现,其中列表中  姓名字符串  id都不一样,但是 年龄列表id却都相同。
  这是因为:python中字符串不可以修改,所以在为tom和anny重新命名的时候,会重新创建一个’tom’和’anny’对象,替换旧的’jack’对象。
  这就说明了,浅复制(shallow copy),它复制了对象,但对于对象中的元素,依然使用引用.



"""浅copy"""
import copy
aa = [1,2,3]
bb = copy.copy(aa)
print id(aa)    #11533088
print id(bb)    #12014776
bb[0] =100
print bb        #[100, 2, 3]
print aa        #[1,2,3]
#由于数字不可变,修改的时候会替换旧的对象
print [id(x) for x in bb]   #[10247196, 10246388, 10246376]
print [id(y) for y in aa]   #[10246400, 10246388, 10246376]
  下面试试对象中可变元素:



lis = [['a'],[1,2],['z',23]]
copyLis = copy.copy(lis)
copyLis[1].append('bar')
print copyLis   #[['a'], [1, 2, 'bar'], ['z', 23]]
print lis       #[['a'], [1, 2, 'bar'], ['z', 23]]
  如果希望复制一个容器对象,以及它里面的所有元素(包含元素的子元素),使用copy.deepcopy,这个方法会消耗一些时间和空间,不过,如果你需要完全复制,这是唯一的方法.



"""深copy"""
deepLis = copy.deepcopy(lis)
deepLis[1].append('foo')   
print deepLis   #[['a'], [1, 2,'foo'], ['z', 23]]
print lis       #[['a'], [1, 2], ['z', 23]]
  
注意:
  1、对于非容器类型(如数字、字符串、和其他‘原子’类型的对象)没有被拷贝一说。
  2、如果元祖变量只包含原子类型对象,则不能深copy。

  参考:http://blog.iyunv.com/sharkw/article/details/1934090
  http://www.01happy.com/python-shallow-copy-and-deep-copy/
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-60559-1-1.html 上篇帖子: 讨论python的数据类型的区别 下篇帖子: Python学习笔记(五):Interprocess Communication
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表