Python中的引用和拷贝浅析

Python中的引用和拷贝浅析,第1张

概述Ifanobject\'svaluecanbemodified,theobjectissaidtobemutable.Ifthevaluecannotbemodified,theobjectissaidtobeimmutable.

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#中的值类型);

一、引用和拷贝(references and copIEs)

当程序中使用=赋值 *** 作符时,例如a=b,

对于不可变的对象,a作为b的一个拷贝被创建,a和b将指向不同的内存地址,a和b相互独立。
复制代码 代码如下:
def Testcopy():
    a = 10
    b = a
    a =20
    print (b) #b still is 10

但是对于可变的对象,a作为b的一个引用被创建,a和b的元素公用相同的内存地址,a和b的元素共享。
复制代码 代码如下:
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,-100,4]

二、深拷贝和浅拷贝(shallow copy and deep copy)

为了避免可变对象指向同一个对象,必须创建一个新的拷贝,而不是引用。
在python中可以对容器对象(例如Lists和dictionarIEs)使用两种拷贝:浅拷贝和深拷贝。
 
浅拷贝创建一个新的对象,但是使用原来对象的元素的引用(如果是不变类型,相当于是拷贝)来填充新对象。可以使用copy.copy()来实现浅拷贝。
复制代码 代码如下:
def TestShallowcopy():
    a = [ 1,[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()来实现深拷贝。

复制代码 代码如下:
def TestDeepcopy():
  import copy
  a = [1,4]]
  b = copy.deepcopy(a)
  b[2][0] = -100
  print (b)  # b is changed
  print (a)  # a is unchanged

在这个例子中,a和b是对立的List对象,且他们的元素也相互独立。

三、引用计数和垃圾回收

python中的所有的对象都是引用计数的,一个对象赋值或加入容器时,它的引用计数就会自增,当使用del时或变量赋值为其他值时,引用计数就会自减,当引用计数为0时,python的垃圾回收器就会回收该变量。
复制代码 代码如下:
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()

运行结果为:

复制代码 代码如下:
11
12
13
14
13
12
11

为啥一上来就有11个引用了呢?谁知道?

您可能感兴趣的文章:解析Python中的变量、引用、拷贝和作用域的问题python实现同时给多个变量赋值的方法深入理解python中函数传递参数是值传递还是引用传递浅谈Python浅拷贝、深拷贝及引用机制Python中实现变量赋值传递时的引用和拷贝方法 总结

以上是内存溢出为你收集整理的Python中的引用和拷贝浅析全部内容,希望文章能够帮你解决Python中的引用和拷贝浅析所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://www.outofmemory.cn/langs/1204359.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-04
下一篇 2022-06-04

发表评论

登录后才能评论

评论列表(0条)

保存