深浅拷贝的补充

Python当中对于拷贝,分为两种类型。一种是数字和字符串,另一种就是列表、元组、字典等其他类型了。

一、数字和字符串的拷贝

1、赋值

举个栗子:

a1 = 123123
a2 = 123123
# a2 = a1  # 赋值
print(id(a1))  # 通过id()函数来打印变量在内存当中的地址
print(id(a2))

输出结果是:

1959780298352
1959780298352 

  在以上代码块当中,a2与a1所赋的值是一样的,都是数字123123。因为python有一个重用机制,对于同一个数字,python并不会开辟一块新的内存空间,而是维护同一块内存地址,只是将该数字对应的内存地址的引用赋值给变量a1和a2。所以根据输出结果,a1和a2其实对应的是同一块内存地址,只是两个不同的引用罢了。同样的,对于a2 = a1,其实效果等同于“a1 = 123123; a2 = 123123”,它也就是将a1指向123123的引用赋值给a2。字符串跟数字的原理雷同,如果把123123改成“abcabc”也是一样的。

结论:对于通过用 = 号赋值,数字和字符串在内存当中用的都是同一块地址。

2、浅拷贝

import copy  # 使用浅拷贝需要导入copy模块

a1 = 123123
a3 = copy.copy(a1)  # 使用copy模块里的copy()函数就是浅拷贝了
print(id(a1))
print(id(a3))

输出结果是:

35233168
35233168

  通过使用copy模块里的copy()函数来进行浅拷贝,把a1拷贝一份赋值给a3,查看输出结果发现,a1和a3的内存地址还是一样。

结论:对于浅拷贝,数字和字符串在内存当中用的也是同一块地址。

3、深拷贝

举个栗子:

import copy

a1 = 123123
a4 = copy.deepcopy(a1)  # 深拷贝是用copy模块里的deepcopy()函数
print(id(a1))
print(id(a4))

输出结果为:

31432080
31432080

  查看结果发现,对于深拷贝,数字和字符串在内存当中用的也是同一块地址。

所以综上所述,对于数字和字符串的赋值、浅拷贝、深拷贝在内存当中用的都是同一块地址。原理如下图:

二、字典、列表、元组等其他类型的拷贝

1、赋值

举个栗子:

n1 = {"k1": "wu", "k2": 123, "k3": ["alex", 678]}
n2 = n1  # 赋值
print(id(n1))
print(id(n2))

输出结果:

2235551677536
2235551677536

  我们的栗子当中用了一个字典n1,字典里面嵌套了一个列表,当我们把n1赋值给n2时,内存地址并没有发生变化,因为其实它也是只是把n1的引用拿过来赋值给n2而已。(我们用了一个字典来举例,其他类型也是一样的)

原理如下图:

结论:对于赋值,字典、列表、元组等其他类型用的内存地址不会变化。

2、浅拷贝

举个栗子:

import copy

n1 = {"k1": "wu", "k2": 123, "k3": ["alex", 678]}
n3 = copy.copy(n1)  # 浅拷贝
print("第一层字典的内存地址:")
print(id(n1))
print(id(n3))
print("第二层嵌套的列表的内存地址:")
print(id(n1["k3"]))
print(id(n3["k3"]))

输出结果:

第一层字典的内存地址:
6516024
6516096
第二层嵌套的列表的内存地址:
36995720
36995720

  通过以上结果可以看出,进行浅拷贝时,我们的字典第一层n1和n3指向的内存地址已经改变了,但是对于第二层里的列表并没有拷贝,它的内存地址还是一样的。原理如下图:

 结论:所以对于浅拷贝,字典、列表、元组等类型,它们只拷贝第一层地址。

3、深拷贝

举个栗子:

import copy

n1 = {"k1": "wu", "k2": 123, "k3": ["alex", 678]}
n4 = copy.deepcopy(n1)  # 深拷贝
print("第一层字典的内存地址:")
print(id(n1))
print(id(n4))
print("第二层嵌套的列表的内存地址:")
print(id(n1["k3"]))
print(id(n4["k3"]))

输出结果:

第一层字典的内存地址:
31157560
35463600
第二层嵌套的列表的内存地址:
35947144
35947336

  通过以上结果发现,进行深拷贝时,字典里面的第一层和里面嵌套的地址都已经变了。对于深拷贝,它会拷贝多层,将第二层的列表也拷贝一份,如果还有第三层嵌套,那么第三层的也会拷贝,但是对于里面的最小元素,比如数字和字符串,这里就是“wu”,123,“alex”,678之类的,按照python的机制,它们会共同指向同一个位置,它的内存地址是不会变的。原理如下图:

结论:对于深拷贝,字典、列表、元组等类型,它里面嵌套多少层,就会拷贝多少层出来,但是最底层的数字和字符串地址不变。

举个实际应用场景的栗子。

我们在维护服务器信息的时候,经常会要更新服务器信息,这时我们重新一个一个添加是比较麻烦的,我们可以把原数据类型拷贝一份,在它的基础上做修改。

栗子一、使用浅拷贝

import copy

dic = {
    "cpu": [80, ],
    "mem": [80, ],
    "disk": [80, ]
}
# 定义了一个字典,存储服务器信息。
print(‘before‘, dic)
new_dic = copy.copy(dic)
new_dic[‘cpu‘][0] = 50  # 更新cpu为50
print(dic)
print(new_dic)

输出结果为:

before {‘cpu‘: [80], ‘mem‘: [80], ‘disk‘: [80]}
{‘cpu‘: [50], ‘mem‘: [80], ‘disk‘: [80]}
{‘cpu‘: [50], ‘mem‘: [80], ‘disk‘: [80]}

这时我们会发现,使用浅拷贝时,我们修改新的字典的值之后,原来的字典里面的cpu值也被修改了,这并不是我们希望看到的。

栗子二、使用深拷贝

import copy

dic = {
    "cpu": [80, ],
    "mem": [80, ],
    "disk": [80, ]
}
print(‘before‘, dic)
new_dic = copy.deepcopy(dic)
new_dic[‘cpu‘][0] = 50
print(dic)
print(new_dic)

输出结果:

before {‘cpu‘: [80], ‘mem‘: [80], ‘disk‘: [80]}
{‘cpu‘: [80], ‘mem‘: [80], ‘disk‘: [80]}
{‘cpu‘: [50], ‘mem‘: [80], ‘disk‘: [80]}

使用深拷贝的时候,发现只有新的字典的cpu值被修改了,原来的字典里面的cpu值没有变。大功告成!

原文地址:https://www.cnblogs.com/eric818/p/10301270.html

时间: 2024-11-02 15:01:39

深浅拷贝的补充的相关文章

深浅拷贝 知识点补充

深浅拷贝 lst1 = ["金毛狮王", "紫衫龙王", "白眉鹰王", "青翼蝠王"] lst2 = lst1 print(lst1) print(lst2) lst1.append("杨逍") print(lst1) print(lst2) 结果: ['金毛狮王', '紫衫龙王', '白眉鹰王', '青翼蝠王', '杨逍'] ['金毛狮王', '紫衫龙王', '白眉鹰王', '青翼蝠王', '杨逍']

7.set集合,深浅拷贝及补充

本节主要内容:1. 基础数据类型补充2. set集合3. 深浅拷?贝主要内容:?一. 基础数据类型补充?首先关于int和str在之前的学习中已经讲了了80%以上了了. 所以剩下的?自?己看?一看就可以了了.我们补充给?一个字符串串基本操作 li = ["李李嘉诚", "麻花藤", "?黄海?峰", "刘嘉玲"] s = "_".join(li) print(s) li = "?黄花?大闺?女女&q

3.python基础补充(集合,collection系列,深浅拷贝)

一.集合 1.集合(set): 把不同的元素组成一起形成集合,是python基本的数据类型.集合元素(set elements):组成集合的成员 python的set和其他语言类似, 是一个无序不重复元素集, 基本功能包括关系测试和消除重复元素. 集合对象还支持union(联合), intersection(交), difference(差)和sysmmetric difference(对称差集)等数学运算. sets 支持 x in set, len(set),和 for x in set.作

知识点补充 set 深浅拷贝

一 对前面知识点的补充 1.str中的join()方法是将列表转换成字符串 lst = ["韩雪","赵丽颖","黄渤","李连杰"] print("_".join(lst)) # 韩雪_赵丽颖_黄渤_李连杰 将每个元素用_进行拼接 s = "马化腾" print("-".join(s)) # 马-化-腾 2.把字符串转换成列表  用split() ss = &quo

python深浅拷贝,集合以及数据类型的补充

1.基础数据类型的补充 1.元组 如果元组中只有一个数据,且没有逗号,则该'元组'与里面的数据的类型相同.如: 1 tu = (1) 2 tu1 = (1,) 3 tu2 = ('alex') 4 tu3 = ([1,2,3],) 5 print(tu,type(tu)) # 1 <class 'int'> 6 print(tu1,type(tu1)) # (1,) <class 'tuple'> 7 print(tu2,type(tu2)) # alex <class 's

深浅拷贝和基础类型补充

1. 基础数据类型补充 大多数的基本数据类型的知识.已经学完了 join() "*".join("马虎疼") # 马*虎*疼 把传递进去的参数进行迭代. 获取到的每个元素和前面的*进行拼接. 得到的是字符串 split() 切割. 切割的结果是列表 列表和字典: 都不能在循环的时候直接删除 把要删除的内容记录在新列表中然后循环这个新列表. 删除列表(字典) fromkeys() 坑1: 返回新字典. 不会更改老字典 坑2: 当value是可变的数据类型. 各个key

巨蟒python全栈开发-第7天 基本数据类型补充&amp;深浅拷贝

1.基本数据类型补充 2.深浅拷贝 DAY5-基本数据类型(基本数据类型补充&深浅拷贝) 本节主要内容: 1 (1)#1.'''#字符串是不可变的数据类型lst=['alex','dsb','wusir','xsb']#大烧饼,小烧饼s='_'.join(lst) #使用前面的字符串,对后面的列表进行拼接,拼接的结果是一个字符串print(s)'''(2)'''#split() 根据你给的参数进行切割,切割的结果是列表 s='alex_dbs_wusir_xsb'lst=s.split("

python基础(9):基本数据类型四(set集合)、基础数据类型补充、深浅拷贝

1. 基础数据类型补充 li = ["李嘉诚", "麻花藤", "?海峰", "刘嘉玲"] s = "_".join(li) print(s) li = "?花?闺?" s = "_".join(li) print(s) 列表: 循环删除列表中的每?个元素 li = [11, 22, 33, 44] for e in li: li.remove(e) print(li

python学习笔记4:基础(集合,collection系列,深浅拷贝)

转载至:http://www.cnblogs.com/liu-yao/p/5146505.html 一.集合 1.集合(set): 把不同的元素组成一起形成集合,是python基本的数据类型.集合元素(set elements):组成集合的成员 python的set和其他语言类似, 是一个无序不重复元素集, 基本功能包括关系测试和消除重复元素. 集合对象还支持union(联合), intersection(交), difference(差)和sysmmetric difference(对称差集)