总结:Python的赋值、深拷贝、浅拷贝有什么区别

自我总结:

1)赋值:对象赋值实际上是对象的引用。

在Python中,变量就是地址的一种表示形式,并不开辟开辟存储空间。

2)浅拷贝:只拷贝了顶层(第一层),没有拷贝子对象。所以子对象的原始数据改变,子对象会改变。

3)深拷贝:区别于浅拷贝只拷贝顶层引用,深拷贝会逐层进行拷贝,直到拷贝的所有引用都是不可变引用为止。

为什么Python默认的拷贝方式是浅拷贝?

时间角度:浅拷贝花费时间更少;

空间角度:浅拷贝花费内存更少;

效率角度:浅拷贝只拷贝顶层数据,一般情况下比深拷贝效率高。

1.python的赋值,深拷贝和浅拷贝的区别

在python中,对象赋值实际上是对象的引用。当创建一个对象,然后把它赋给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用。

(1)直接赋值,默认浅拷贝传递对象的引用而已,原始列表改变,被赋值的b也会做相同的改变

>>> b=alist
>>> print b
[1, 2, 3, [‘a‘, ‘b‘]]
>>> alist.append(5)
>>> print alist;print b
[1, 2, 3, [‘a‘, ‘b‘], 5]
[1, 2, 3, [‘a‘, ‘b‘], 5]

(2)copy浅拷贝,没有拷贝子对象,所以原始数据改变,子对象会改变

>>> import copy

>>> c=copy.copy(alist)
>>> print alist;print c
[1, 2, 3, [‘a‘, ‘b‘]]
[1, 2, 3, [‘a‘, ‘b‘]]
>>> alist.append(5)
>>> print alist;print c
[1, 2, 3, [‘a‘, ‘b‘], 5]
[1, 2, 3, [‘a‘, ‘b‘]]

>>> alist[3]
[‘a‘, ‘b‘]
>>> alist[3].append(‘cccc‘)
>>> print alist;print c
[1, 2, 3, [‘a‘, ‘b‘, ‘cccc‘], 5]
[1, 2, 3, [‘a‘, ‘b‘, ‘cccc‘]] 里面的子对象被改变了

深拷贝,包含对象里面的自对象的拷贝,所以原始对象的改变不会造成深拷贝里任何子元素的改变

>>> import copy

>>> d=copy.deepcopy(alist)
>>> print alist;print d
[1, 2, 3, [‘a‘, ‘b‘]]
[1, 2, 3, [‘a‘, ‘b‘]]始终没有改变
>>> alist.append(5)
>>> print alist;print d
[1, 2, 3, [‘a‘, ‘b‘], 5]
[1, 2, 3, [‘a‘, ‘b‘]]始终没有改变
>>> alist[3]
[‘a‘, ‘b‘]
>>> alist[3].append("ccccc")
>>> print alist;print d
[1, 2, 3, [‘a‘, ‘b‘, ‘ccccc‘], 5]
[1, 2, 3, [‘a‘, ‘b‘]]  始终没有改变

2. python的复制,深拷贝和浅拷贝的背景与意义?

不像matlab一样,比如b=a,就是单纯用a给b赋值,之后a怎么变,b不会变化。c语言也是一样。

如果想a怎么变,b就怎么变,c语言里就提出了引用的概念,相当于别名了。

example:

int a; int &ra=a; //定义引用ra,它是变量a的引用,即别名

okay!这篇文章有详解浅拷贝与深拷贝的本质原因:重新开辟内存来存储

https://www.jianshu.com/p/9ed9b5ce7bb0

https://baijiahao.baidu.com/s?id=1627356407968660842&wfr=spider&for=pc

这篇文章提出pythond的这个设计可以防止数据篡改,或者灵活改变

在Python中,对对象有一种很通俗的说法,万物皆对象。说的就是构造的任何数据类型都是一个对象,无论是数字、字符串、还是函数,甚至是模块、Python都对当做对象处理。

所有Python对象都拥有三个属性:身份、类型、值。

  1. name="Li J"

  2.  

    print(type(name))

  3.  

    print(id(name))

  4.  

    print(name)

  5.  

  6.  

    #输出:

  7.  

    #<type ‘str‘>

  8.  

    #140334394101408

  9.  

    #Li J

可变与不可变对象

在Python中,按更新对象的方式,可以将对象分为2大类:可变对象与不可变对象。

可变对象: 列表、字典、集合。所谓可变是指可变对象的值可变,身份是不变的。

不可变对象:数字、字符串、元组。不可变对象就是对象的身份和值都不可变。新创建的对象被关联到原来的变量名,旧对象被丢弃,垃圾回收器会在适当的时机回收这些对象。

  1. var1="python" #字符串类型是不可变的

  2.  

    print(id(var1))

  3.  

    var1="java"

  4.  

    print(id(var1))

  5.  

  6.  

    a=[3,4] #list是可变的,

  7.  

    print(id(a))

  8.  

    a.append(3)

  9.  

    print(id(a))

  10.  

  11.  

    #输出结果:

  12.  

    140591145210096

  13.  

    140591145211632

  14.  

    140590909362688

  15.  

    140590909362688

引用

在Python程序中,每个对象都会在内存中申请开辟一块空间来保存该对象,该对象在内存中所在位置的地址被称为引用。在开发程序时,所定义的变量名实际就对象的地址引用。

引用实际就是内存中的一个数字地址编号,在使用对象时,只要知道这个对象的地址,就可以操作这个对象,但是因为这个数字地址不方便在开发时使用和记忆,所以使用变量名的形式来代替对象的数字地址。在Python中,变量就是地址的一种表示形式,并不开辟开辟存储空间。

就像 IP 地址,在访问网站时,实际都是通过 IP 地址来确定主机,而 IP 地址不方便记忆,所以使用域名来代替 IP 地址,在使用域名访问网站时,域名被解析成 IP 地址来使用。

通过一个例子来说明变量和变量指向的引用就是一个东西:

  1. b=18

  2.  

    print(id(b))

  3.  

    print(id(18))

  4.  

  5.  

    输出:

  6.  

    29413312

  7.  

    29413312

浅拷贝:

  1. print("浅拷贝:")

  2.  

    import copy

  3.  

    b=[1,2,3,4,5]

  4.  

    print("id b:",id(b))

  5.  

    h=copy.copy(b)

  6.  

    print("id h",id(h))

  7.  

    print(h)

  8.  

    h.append(6)

  9.  

    print(h)

  10.  

    print("id h",id(h))

  11.  

    print(b) #浅拷贝新的列表h改变了,原来的b没变。

  12.  

  13.  

    b[1]=‘n‘ #列表元素改变后,新的列表也没变

  14.  

    print(h)

  15.  

  16.  

    输出:

  17.  

    浅拷贝:

  18.  

    (‘id b:‘, 140165805110552)

  19.  

    (‘id h‘, 140165805110480)

  20.  

    [1, 2, 3, 4, 5]

  21.  

    [1, 2, 3, 4, 5, 6]

  22.  

    (‘id h‘, 140165805110480)

  23.  

    [1, 2, 3, 4, 5]

  24.  

    [1, 2, 3, 4, 5, 6]

  1. a = [1, 2]

  2.  

    l1 = [3, 4, a]

  3.  

    l2 = copy.copy(l1)

  4.  

    print(l1)

  5.  

    print(l2)

  6.  

    print(id(l1))

  7.  

    print(id(l2))

  8.  

    a[0] = 11

  9.  

  10.  

    print(id(l1))

  11.  

    print(id(l2))

  12.  

    print(l1)

  13.  

    print(l2)

  14.  

    输出:

  15.  

    [3, 4, [1, 2]]

  16.  

    [3, 4, [1, 2]]

  17.  

    140624327425704

  18.  

    140624326197400

  19.  

    140624327425704

  20.  

    140624326197400

  21.  

    [3, 4, [11, 2]]

  22.  

    [3, 4, [11, 2]]

可以看出浅拷贝,相当于只拷贝了一层,到a那里,a变化了,其值也就变化了。

Python中有多种方式实现浅拷贝,copy模块的copy函数、对象的copy函数、工厂方法、切片等;大多数情况下,编写程序时都是使用浅拷贝,除非有特定的需求;浅拷贝的优点:拷贝速度快,占用空间少,拷贝效率高。

深拷贝

区别于浅拷贝只拷贝顶层引用,深拷贝会逐层进行拷贝,直到拷贝的所有引用都是不可变引用为止。

  1. a = [1, 2]

  2.  

    l1 = [3, 4, a]

  3.  

    l2 = copy.deepcopy(l1)

  4.  

    print(l1)

  5.  

    print(l2)

  6.  

    print(id(l1))

  7.  

    print(id(l2))

  8.  

    a[0] = 11

  9.  

  10.  

    print(id(l1))

  11.  

    print(id(l2))

  12.  

    print(l1)

  13.  

    print(l2)

  14.  

  15.  

    输出:

  16.  

    [3, 4, [1, 2]]

  17.  

    [3, 4, [1, 2]]

  18.  

    140673014398488

  19.  

    140672779715720

  20.  

    140673014398488

  21.  

    140672779715720

  22.  

    [3, 4, [11, 2]]

  23.  

    [3, 4, [1, 2]]

参考文献

https://www.jianshu.com/p/9ed9b5ce7bb0

https://baijiahao.baidu.com/s?id=1627356407968660842&wfr=spider&for=pc

原文地址:https://www.cnblogs.com/zhaoyingzhe/p/12394630.html

时间: 2024-10-11 22:53:53

总结:Python的赋值、深拷贝、浅拷贝有什么区别的相关文章

Python 直接赋值、浅拷贝和深度拷贝区别

Python 直接赋值.浅拷贝和深度拷贝区别 转自https://www.runoob.com/w3cnote/python-understanding-dict-copy-shallow-or-deep.html 直接赋值:其实就是对象的引用(别名). 浅拷贝(copy):拷贝父对象,不会拷贝对象的内部的子对象. 深拷贝(deepcopy): copy 模块的 deepcopy 方法,完全拷贝了父对象及其子对象. 解析 1.b = a: 赋值引用,a 和 b 都指向同一个对象. # -*- c

Python中赋值、浅拷贝与深拷贝

python中关于对象复制有三种类型的使用方式,赋值.浅拷贝与深拷贝.他们既有区别又有联系,刚好最近碰到这一类的问题,研究下. 一.赋值 在python中,对象的赋值就是简单的对象引用,这点和C++不同.如下: list_a = [1,2,3,"hello",["python","C++"]] list_b = list_a 这种情况下,list_b和list_a是一样的,他们指向同一片内存,list_b不过是list_a的别名,是引用. 我们可

关于python中赋值、浅拷贝、深拷贝之间区别的深入分析

大家都知道,在python中复制一个对象有多种方法,其中常用的是赋值.浅拷贝和深拷贝,这三者之间有哪些区别和哪些坑呢? 首先,定义一下: 赋值:  a =1    b =a    a赋值给了b 浅拷贝: a = []  b = a.copy() 或者import copy             b = copy.copy(a) 深拷贝:import copy  a = []   b = copy.deepcopy(a) 未完待续

Python对象赋值、浅拷贝、深拷贝

Python中,基本数据类型,理解为常见数据类型:布尔型.整型.浮点型.字符串.列表.元组.字典.集合,随语言不同而不同,但是根据在内存中存储方式的不同,区分开原子类型和容器类型. 对象赋值 对象的赋值都是进行(对象引用传递)/(内存地址传递)/(内存引用),所以当一个对象改变,另一个同步改变. 结合代码思考 ··· will = ["Will", 28, ["Python", "C#", "JavaScript"]] # 元

python中赋值,浅拷贝,深拷贝的区别

1.首先,对被操作对象分类,对于不可变对象而言,如字符串.数字.tuple等,这三种操作是等价的,都是引用 import copy a='apple'b=ac=copy.copy(a)d=copy.deepcopy(a)print(id(a))print(id(b))print(id(c))print(id(d) 输出: 1840734496184073449618407344961840734496 可见,这四个变量都指向同一块内存地址,即'apple'这个字符串所在的地址 2.对可变对象(或

python直接赋值、浅拷贝与深拷贝的区别解析

直接赋值:其实就是对象的引用(别名). 浅拷贝(copy):拷贝父对象,不会拷贝对象的内部的子对象. 深拷贝(deepcopy): copy 模块的 deepcopy 方法,完全拷贝了父对象及其子对象. 实例解析 a = {1: [1,2,3]} 1. b = a: 赋值引用,a 和 b 都指向同一个对象,如下图:  2. b = a.copy(): 浅拷贝, a 和 b 是一个独立的对象,但他们的子对象还是指向统一对象(是引用),如下图:  3. b = copy.deepcopy(a): 需

Python 直接赋值、浅拷贝和深度拷贝全解析

直接赋值:其实就是对象的引用(别名). 浅拷贝(copy):拷贝父对象,不会拷贝对象的内部的子对象. 深拷贝(deepcopy): copy 模块的 deepcopy 方法,完全拷贝了父对象及其子对象. 字典浅拷贝实例 实例 1 2 3 4 5 6 7 >>>a = {1: [1,2,3]} >>> b = a.copy() >>> a, b ({1: [1, 2, 3]}, {1: [1, 2, 3]}) >>> a[1].appe

Python 直接赋值、浅拷贝和深度拷贝解析

直接赋值:其实就是对象的引用(别名). 浅拷贝(copy):拷贝父对象,不会拷贝对象的内部的子对象. 深拷贝(deepcopy): copy 模块的 deepcopy 方法,完全拷贝了父对象及其子对象. 字典浅拷贝实例 实例 >>> a = {1: [1,2,3]} >>> b = a.copy() >>> a, b ({1: [1, 2, 3]}, {1: [1, 2, 3]}) >>> a[1].append(4) >>

Python直接赋值、浅拷贝和深度拷贝解析

ython 文件 writelines() 方法用于向文件中写入一序列的字符串. 这一序列字符串可以是由迭代对象产生的,如一个字符串列表. 换行需要制定换行符 \n. 语法 writelines() 方法语法如下: 1fileObject.writelines(iterable)参数 iterable -- 可迭代对象(字符串.列表.元祖.字典). 返回值 该方法没有返回值. 原文地址:http://blog.51cto.com/98405/2145034

python中关于赋值、浅拷贝与深拷贝的问题

Python中关于对象复制有三种类型的使用方式,赋值.浅拷贝与深拷贝.他们既有区别又有联系,刚好最近碰到这一类的问题,研究下. 一.赋值 在python中,对象的赋值就是简单的对象引用,这点和C++不同.如下: list_a = [1,2,3,"hello",["python","C++"]] list_b = list_a 这种情况下,list_b和list_a是一样的,他们指向同一片内存,list_b不过是list_a的别名,是引用. 我们可