《Python核心编程》读书笔记之Python对象浅拷贝和深拷贝

Python里的对象赋值实际上是简单的对象引用。也就是说,当你创建一个对象,然后把它赋给另一个变量的时候,Python并没有拷贝这个对象,而是拷贝了这个对象的引用。我们使用id()内建函数来显示每个对象的标识符。

>>> person = ['name', ['age', 24]]
>>> jack = person
>>> [id(x) for x in [person, jack]]
[140562141461640, 140562141461640]
对一个对象进行浅拷贝其实是新创建了一个类型跟原对象一样,其内容是原来对象元素的引用,换句话说,这个拷贝的对象本身是新的,但是它的内容不是。序列类型对象的浅拷贝是默认类型拷贝,并可以以下几种方式实施(1)完全切片操作[:];(2)利用工厂函数,比如list()、dict()等;(3)使用copy模块的copy函数。

>>> person = ['name', ['age', 24]]
>>> tom = person[:]
>>> jerry = list(person)
>>> [id(x) for x in [person, tom, jerry]]
[140562141554376, 140562141552776, 140562141554760]
>>> tom[0] = 'tom'
>>> jerry[0] = 'jerry'
>>> tom, jerry
(['tom', ['age', 24]], ['jerry', ['age', 24]])
>>> tom[1][1] = 12
>>> tom, jerry
(['tom', ['age', 12]], ['jerry', ['age', 12]])
你可能会问,当jerry的名字被赋值,为什么tom的名字没有受到影响?这是因为在这两个列表的两个对象中,第一个对象是不可变的(字符串类型),而第二个是可变的(列表)。正因如此,当进行浅拷贝时,字符串被显式地拷贝,并新创建了一个字符串对象,而列表元素只是把它的引用复制了一下,并不是它的成员。

改变前

>>> [id(x) for x in tom]
[140562171833232, 140562141552840]
>>> [id(x) for x in jerry]
[140562171833232, 140562141552840]
改变后

>>> [id(x) for x in tom]
[140562141557008, 140562141552840]
>>> [id(x) for x in jerry]
[140562141557120, 140562141552840]
要得到一个完全拷贝或者说深拷贝——创建一个新的容器对象,包含原有对象元素(引用)全新拷贝的引用——需要copy.deepcopy()函数。

>>> person = ['name', ['age', 24]]
>>> tom = person
>>> import copy
>>> jerry = copy.deepcopy(person)
>>> [id(x) for x in [person, tom, jerry]]
[140562141554568, 140562141554568, 140562141553800]
>>> tom[0] = 'tom'
>>> jerry[0] = 'jerry'
>>> tom, jerry
(['tom', ['age', 24]], ['jerry', ['age', 24]])
>>> tom[1][1] = 12
>>> tom, jerry
(['tom', ['age', 12]], ['jerry', ['age', 24]])
>>> [id(x) for x in tom]
[140562141557008, 140562141555144]
>>> [id(x) for x in jerry]
[140562141557120, 140562129735432]
以下有几点关于拷贝操作的警告。第一,非容器类型(比如数字、字符串和其他“原子”类型的对象,像代码、类型和xrange对象等)没有被拷贝一说,浅拷贝是用完全切片操作来完成的。第二,如果元组变量只包含原子类型的对象,对它的深拷贝将不会进行。

>>> person = ['name', ('age', 24)]
>>> newPerson = copy.deepcopy(person)
>>> [id(x) for x in [person, newPerson]]
[140562141555080, 140562129842760]
>>> [id(x) for x in person]
[140562171833232, 140562141548488]
>>> [id(x) for x in newPerson]
[140562171833232, 140562141548488]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值