详解Python中列表切片及浅拷贝的关系(下)_对列表操作时为什么要浅拷贝

最后

不知道你们用的什么环境,我一般都是用的Python3.6环境和pycharm解释器,没有软件,或者没有资料,没人解答问题,都可以免费领取(包括今天的代码),过几天我还会做个视频教程出来,有需要也可以领取~

给大家准备的学习资料包括但不限于:

Python 环境、pycharm编辑器/永久激活/翻译插件

python 零基础视频教程

Python 界面开发实战教程

Python 爬虫实战教程

Python 数据分析实战教程

python 游戏开发实战教程

Python 电子书100本

Python 学习路线规划

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!


代码输出如下:



0x7ffeea02a190
0x7ffeea02a190
0x7ffeea02a190
0x7ffeea02a2b0
[1, 2, 3, [8, 5]]
[10, 2, 3, [8, 5]]


可以看到,对于外层元素,两个列表之间实现了数据隔离,而对于嵌套的复合对象,两个列表之间仍然是共享的。我们将上述代码转化成内存示意图,就能更直观地看出在拷贝以及修改过程中究竟发生了什么:


![图片](https://img-blog.csdnimg.cn/9306b156f6984fd9a408e024678cd663.png)


可以看到,浅拷贝在拷贝复杂对象时仍然采取了共享策略,于是,对嵌套列表的修改导致list01、list02的数据都发生了变化。


当我们要对列表中嵌套的列表保持数据隔离时,深拷贝就派上用场了。在Python中,实现深拷贝的方法是调用copy中的deepcopy方法,请看如下代码:



import copy

list01 = [1, 2, 3, [4, 5]]
print(“----------------使用深拷贝的方法复制一个list02----------------”)
list02 = copy.deepcopy(list01)
print(“list01 第一个元素的地址是:{}”.format(hex(id(list01[0]))))
print(“list01 中嵌套的列表地址是:{}”.format(hex(id(list01[3]))))
print(“list01 中嵌套的列表的第一个元素地址是:{}”.format(hex(id(list01[3][0]))))
print(“list02 第一个元素的地址是:{}”.format(hex(id(list02[0]))))
print(“list02 中嵌套的列表地址是:{}”.format(hex(id(list02[3]))))
print(“list02 中嵌套的列表的第一个元素地址是:{}”.format(hex(id(list02[3][0]))))
print(“-----------------下面对list02中的元素进行修改-----------------”)
list02[0] = 10
list02[3][0] = 8
print(“list01 第一个元素的地址是:{}”.format(hex(id(list01[0]))))
print(“list01 中嵌套的列表地址是:{}”.format(hex(id(list01[3]))))
print(“list01 中嵌套的列表的第一个元素地址是:{}”.format(hex(id(list01[3][0]))))
print(“list02 第一个元素的地址是:{}”.format(hex(id(list02[0]))))
print(“list02 中嵌套的列表地址是:{}”.format(hex(id(list02[3]))))
print(“list02 中嵌套的列表的第一个元素地址是:{}”.format(hex(id(list02[3][0]))))
print(“list01的内容是:{}”.format(list01))
print(“list02的内容是:{}”.format(list02))



> 
> 如果对python自动化测试、web自动化、接口自动化、移动端自动化、面试经验交流等等感兴趣的测试人,**[可以 点这自行获取…]( )**
> 
> 
> 


上述代码的输出如下:



----------------使用深拷贝的方法复制一个list02----------------
list01 第一个元素的地址是:0x7ffee9d9a190
list01 中嵌套的列表地址是:0x148f6d95188
list01 中嵌套的列表的第一个元素地址是:0x7ffee9d9a1f0
list02 第一个元素的地址是:0x7ffee9d9a190
list02 中嵌套的列表地址是:0x148f6ecbf08
list02 中嵌套的列表的第一个元素地址是:0x7ffee9d9a1f0
-----------------下面对list02中的元素进行修改-----------------
list01 第一个元素的地址是:0x7ffee9d9a190
list01 中嵌套的列表地址是:0x148f6d95188
list01 中嵌套的列表的第一个元素地址是:0x7ffee9d9a1f0
list02 第一个元素的地址是:0x7ffee9d9a2b0
list02 中嵌套的列表地址是:0x148f6ecbf08
list02 中嵌套的列表的第一个元素地址是:0x7ffee9d9a270
list01的内容是:[1, 2, 3, [4, 5]]
list02的内容是:[10, 2, 3, [8, 5]]


在上述代码的输出中,我们至少可以看出两点:1、对于列表中包含的数值类型,在深拷贝时依然采取了写时复制的策略,也就是说只有当这些元素被修改时才会重新申请空间写入新数据;


2、对于列表中包含的列表,在深拷贝时会逐层拷贝,但嵌套列表中包含的数值类型仍然采取共享策略。


我们用内存图来表达上述深拷贝过程,会更直观更清晰。


![图片](https://img-blog.csdnimg.cn/878f7b42153a444b9bce7f4e8e3fa2f7.png)


### 列表与函数参数传递


在开发过程中经常遇到的一个问题是:列表在函数参数传递时使用的是浅拷贝还是深拷贝?如果你的回答是浅拷贝,那么恭喜你,答错了!


列表在函数参数传递过程中使用的既不是浅拷贝也不是深拷贝,而是简单的引用传递。请看下面的程序:



list01 = [“幻视”, “鹰眼”, “蜘蛛侠”]

def listPrint(strList):
print(hex(id(strList)))
for item in strList:
print(item, end=‘\t’)
print()
strList[0] = “灭霸”
print(“--------------在listPrint函数里面----------------”)
print(strList)

print(hex(id(list01)))
listPrint(list01)
print(“--------------调用完listPrint以后----------------”)
print(list01)


程序的输出如下:



0x234a59b5188
0x234a59b5188
幻视 鹰眼 蜘蛛侠
--------------在listPrint函数里面----------------
[‘灭霸’, ‘鹰眼’, ‘蜘蛛侠’]
--------------调用完listPrint以后----------------
[‘灭霸’, ‘鹰眼’, ‘蜘蛛侠’]


在进行listPrint函数调用时,解释器仅仅是简单地将list01的引用放在了函数listPrint的栈帧上,在函数listPrint中对列表的修改都将影响到list01。


我们在调用listPrint函数时对list01做一次简单的浅拷贝,观察程序输出:



list01 = [“幻视”, “鹰眼”, “蜘蛛侠”]

def listPrint(strList):
print(hex(id(strList)))
for item in strList:
print(item, end=‘\t’)
print()
strList[0] = “灭霸”
print(“--------------在listPrint函数里面----------------”)
print(strList)

print(hex(id(list01)))
listPrint(list01.copy())
print(“--------------调用完listPrint以后----------------”)
print(list01)


上述代码的输出如下:



0x250328b5188
0x250328b5388
幻视 鹰眼 蜘蛛侠
--------------在listPrint函数里面----------------
[‘灭霸’, ‘鹰眼’, ‘蜘蛛侠’]
--------------调用完listPrint以后----------------
[‘幻视’, ‘鹰眼’, ‘蜘蛛侠’]


我们发现,此时listPrint对实参的修改已经不能影响到list01了,因为函数listPrint接收到的参数是list01的一份浅拷贝,而不是list01本身,从而实现了一次简单的数据隔离。


### 深/浅拷贝的拓展


![在这里插入图片描述](https://img-blog.csdnimg.cn/20210511152217670.jpg?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3poaWd1aWd1,size_16,color_FFFFFF,t_70)

**感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的:**



① 2000多本Python电子书(主流和经典的书籍应该都有了)

② Python标准库资料(最全中文版)

③ 项目源码(四五十个有趣且经典的练手项目及源码)

④ Python基础入门、爬虫、web开发、大数据分析方面的视频(适合小白学习)

⑤ Python学习路线图(告别不入流的学习)




**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化学习资料的朋友,可以戳这里获取](https://bbs.csdn.net/forums/4304bb5a486d4c3ab8389e65ecb71ac0)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

  • 28
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值