源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

Python中的引用和拷贝浅析

  • 时间:2022-11-15 00:33 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:Python中的引用和拷贝浅析
If an object's value can be modified, the object is said to be mutable. If the value cannot be modified,the object is said to be immutable. mutable 可变类型,例如 list,set,自定义类型(等价于C#中的引用类型); immutable 不可变类型,例如string,numbers等(等价于C#中的值类型); [b]一、引用和拷贝(references and copies)[/b] 当程序中使用=赋值操作符时,例如a=b, 对于不可变的对象,a作为b的一个拷贝被创建,a和b将指向不同的内存地址,a和b相互独立。
[u]复制代码[/u] 代码如下:
def TestCopy():     a = 10     b = a     a =20     print (b) #b still is 10
但是对于可变的对象,a作为b的一个引用被创建,a和b的元素公用相同的内存地址,a和b的元素共享。
[u]复制代码[/u] 代码如下:
def TestRef():     a=[1,2,3,4]     b=a   #b is a reference to a     print (b is a) # True     b[2] = -100 #change an element in b     print (a) # a also changed to [1,2,-100,4]
[b]二、深拷贝和浅拷贝(shallow copy and deep copy)[/b] 为了避免可变对象指向同一个对象,必须创建一个新的拷贝,而不是引用。 在python中可以对容器对象(例如lists和dictionaries)使用两种拷贝:浅拷贝和深拷贝。   浅拷贝创建一个新的对象,但是使用原来对象的元素的引用(如果是不变类型,相当于是拷贝)来填充新对象。可以使用copy.copy()来实现浅拷贝。
[u]复制代码[/u] 代码如下:
def TestShallowCopy():     a = [ 1, 2, [3,4] ]     b = list(a) # create a shallow copy of a     print (b is a) #False     b.append(100) #append element to b     print (b)     print (a) # a is unchanged     b[2][0] = -100 # modify an element inside b     print (b)     print (a)  # a is changed
在这个例子中,a和b共享相同的可变元素。所以修改其中一个list对象中的元素,另一个list对象也会被修改。 深拷贝创建一个新的对象,同时递归地拷贝对象所包含的所有的元素。可以使用copy.deepcopy()来实现深拷贝。
[u]复制代码[/u] 代码如下:
def TestDeepCopy():   import copy   a = [1, 2, [3, 4]]   b = copy.deepcopy(a)   b[2][0] = -100   print (b)  # b is changed   print (a)  # a is unchanged
在这个例子中,a和b是对立的list对象,且他们的元素也相互独立。 [b]三、引用计数和垃圾回收[/b] python中的所有的对象都是引用计数的,一个对象赋值或加入容器时,它的引用计数就会自增,当使用del时或变量赋值为其他值时,引用计数就会自减,当引用计数为0时,python的垃圾回收器就会回收该变量。
[u]复制代码[/u] 代码如下:
def TestGarbageCollection():   import sys   print(sys.getrefcount(37))   a = 37 # Creates an object with value 37   print(sys.getrefcount(37))   b = a # Increases reference count on 37   print(sys.getrefcount(37))   c = []   c.append(b) # Increases reference count on 37   print(sys.getrefcount(37))   del a # Decrease reference count of 37   print(sys.getrefcount(37))   b = 42 # Decrease reference count of 37   print(sys.getrefcount(37))   c[0] = 2.0 # Decrease reference count of 37   print(sys.getrefcount(37))   TestGarbageCollection()
运行结果为:
[u]复制代码[/u] 代码如下:
11 12 13 14 13 12 11
为啥一上来就有11个引用了呢?谁知道?
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部