python中copy()与deepcopy()的区别小结


Posted in Python onAugust 03, 2018

前言

copy()与deepcopy()之间的区分必须要涉及到python对于数据的存储方式。

深复制被复制对象完全再复制一遍作为独立的新个体单独存在。所以改变原有被复制对象不会对已经复制出来的新对象产生影响。

浅复制并不会产生一个独立的对象单独存在,他只是将原有的数据块打上一个新标签,所以当其中一个标签被改变的时候,数据块就会发生变化,另一个标签也会随之改变。

import copy
 origin = [1, 2, [3, 4]]
#origin 里边有三个元素:1, 2,[3, 4]
cop1 = copy.copy(origin)
cop2 = copy.deepcopy(origin)
cop1 == cop2
------>True
 cop1 is cop2
------>False 
#cop1 和 cop2 看上去相同,但已不再是同一个object
 origin[2][0] = "hey!" 
 origin
------>[1, 2, ['hey!', 4]]
 cop1
------>[1, 2, ['hey!', 4]]
 cop2
------>[1, 2, [3, 4]]

可以看到 cop1,也就是 copy 跟着 origin 改变了。而 cop2 ,也就是 deep copy 并没有变。

Python存储方式

Python 存储变量的方法跟其他 OOP 语言不同。它与其说是把值赋给变量,不如说是给变量建立了一个到具体值的 reference。

当在 Python 中 a = something 应该理解为给 something 贴上了一个标签 a。当再赋值给 a 的时候,就好象把 a 这个标签从原来的 something 上拿下来,贴到其他对象上,建立新的 reference。 这就解释了一些 Python 中可能遇到的诡异情况:

>> a = [1, 2, 3]
>>> b = a
>>> a = [4, 5, 6] //赋新的值给 a
>>> a
[4, 5, 6]
>>> b
[1, 2, 3]
# a 的值改变后,b 并没有随着 a 变

>>> a = [1, 2, 3]
>>> b = a
>>> a[0], a[1], a[2] = 4, 5, 6 //改变原来 list 中的元素
>>> a
[4, 5, 6]
>>> b
[4, 5, 6]
# a 的值改变后,b 随着 a 变了

上面两段代码中,a 的值都发生了变化。区别在于,第一段代码中是直接赋给了 a 新的值(从 [1, 2, 3] 变为 [4, 5, 6]);而第二段则是把 list 中每个元素分别改变。

而对 b 的影响则是不同的,一个没有让 b 的值发生改变,另一个变了。怎么用上边的道理来解释这个诡异的不同呢?

首次把 [1, 2, 3] 看成一个物品。a = [1, 2, 3] 就相当于给这个物品上贴上 a 这个标签。而 b = a 就是给这个物品又贴上了一个 b 的标签。

python中copy()与deepcopy()的区别小结

第一种情况:

a = [4, 5, 6] 就相当于把 a 标签从 [1 ,2, 3] 上撕下来,贴到了 [4, 5, 6] 上。

在这个过程中,[1, 2, 3] 这个物品并没有消失。 b 自始至终都好好的贴在 [1, 2, 3] 上,既然这个 reference 也没有改变过。 b 的值自然不变。

python中copy()与deepcopy()的区别小结

第二种情况:

a[0], a[1], a[2] = 4, 5, 6 则是直接改变了 [1, 2, 3] 这个物品本身。把它内部的每一部分都重新改装了一下。内部改装完毕后,[1, 2, 3] 本身变成了 [4, 5, 6]。

而在此过程当中,a 和 b 都没有动,他们还贴在那个物品上。因此自然 a b 的值都变成了 [4, 5, 6]。

搞明白这个之后就要问了,对于一个复杂对象的浅copy,在copy的时候到底发生了什么?
再看一段代码:

>>> import copy
>>> origin = [1, 2, [3, 4]]
#origin 里边有三个元素:1, 2,[3, 4]
>>> cop1 = copy.copy(origin)
>>> cop2 = copy.deepcopy(origin)
>>> cop1 == cop2
True
>>> cop1 is cop2
False 
#cop1 和 cop2 看上去相同,但已不再是同一个object
>>> origin[2][0] = "hey!" 
>>> origin
[1, 2, ['hey!', 4]]
>>> cop1
[1, 2, ['hey!', 4]]
>>> cop2
[1, 2, [3, 4]]
#把origin内的子list [3, 4] 改掉了一个元素,观察 cop1 和 cop2

学过docker的人应该对镜像这个概念不陌生,我们可以把镜像的概念套用在copy上面。

概念图如下:

python中copy()与deepcopy()的区别小结

copy对于一个复杂对象的子对象并不会完全复制,什么是复杂对象的子对象呢?就比如序列里的嵌套序列,字典里的嵌套序列等都是复杂对象的子对象。对于子对象,python会把它当作一个公共镜像存储起来,所有对他的复制都被当成一个引用,所以说当其中一个引用将镜像改变了之后另一个引用使用镜像的时候镜像已经被改变了。

所以说看这里的origin[2],也就是 [3, 4] 这个 list。根据 shallow copy 的定义,在 cop1[2] 指向的是同一个 list [3, 4]。那么,如果这里我们改变了这个 list,就会导致 origin 和 cop1 同时改变。这就是为什么上边 origin[2][0] = “hey!” 之后,cop1 也随之变成了 [1, 2, [‘hey!', 4]]。

而deepcopy概念图如下:

python中copy()与deepcopy()的区别小结

deepcopy的时候会将复杂对象的每一层复制一个单独的个体出来。

这时候的 origin[2] 和 cop2[2] 虽然值都等于 [3, 4],但已经不是同一个 list了。即我们寻常意义上的复制。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
Python中的二叉树查找算法模块使用指南
Jul 04 Python
Python实现批量更换指定目录下文件扩展名的方法
Sep 19 Python
浅谈Python类的__getitem__和__setitem__特殊方法
Dec 25 Python
完美解决在oj中Python的循环输入问题
Jun 25 Python
python用插值法绘制平滑曲线
Feb 19 Python
Python+OpenCV+图片旋转并用原底色填充新四角的例子
Dec 12 Python
解决Python使用列表副本的问题
Dec 19 Python
Python3实现发送邮件和发送短信验证码功能
Jan 07 Python
python中for in的用法详解
Apr 17 Python
QML实现钟表效果
Jun 02 Python
python框架flask入门之路由及简单实现方法
Jun 07 Python
Python利用zhdate模块实现农历日期处理
Mar 31 Python
Python爬取个人微信朋友信息操作示例
Aug 03 #Python
python opencv人脸检测提取及保存方法
Aug 03 #Python
Python爬虫爬取新浪微博内容示例【基于代理IP】
Aug 03 #Python
OpenCV+python手势识别框架和实例讲解
Aug 03 #Python
Windows下将Python文件打包成.EXE可执行文件的方法
Aug 03 #Python
Python测试网络连通性示例【基于ping】
Aug 03 #Python
python版opencv摄像头人脸实时检测方法
Aug 03 #Python
You might like
PHP文件打开、关闭、写入的判断与执行代码
2011/05/24 PHP
PHP扩展CURL的用法详解
2014/06/20 PHP
thinkphp使用phpmailer发送邮件的方法
2014/11/24 PHP
对PHP PDO的一些认识小结
2015/01/23 PHP
详解PHP数组赋值方法
2015/11/07 PHP
CodeIgniter配置之routes.php用法实例分析
2016/01/19 PHP
PHP版微信第三方实现一键登录及获取用户信息的方法
2016/10/14 PHP
thinkPHP中验证码的简单实现方法
2016/12/05 PHP
php实现的pdo公共类定义与用法示例
2017/07/19 PHP
jQuery中live方法的重复绑定说明
2011/10/21 Javascript
jQuery动画效果animate和scrollTop结合使用实例
2014/04/02 Javascript
jQuery判断checkbox是否选中的3种方法
2014/08/12 Javascript
微信小程序(六):列表上拉加载下拉刷新示例
2017/01/13 Javascript
解决vue2.x中数据渲染以及vuex缓存的问题
2017/07/13 Javascript
Vue组件中slot的用法
2018/01/30 Javascript
JavaScript中创建原子的方法总结
2018/08/26 Javascript
layer.open的自适应及居中及子页面标题的修改方法
2019/09/05 Javascript
ElementUI中el-tree节点的操作的实现
2020/02/27 Javascript
vue 百度地图(vue-baidu-map)绘制方向箭头折线实例代码详解
2020/04/28 Javascript
vue 组件简介
2020/07/31 Javascript
Jquery cookie插件实现原理代码解析
2020/08/04 jQuery
nuxt 路由、过渡特效、中间件的实现代码
2020/11/06 Javascript
如何在JavaScript中等分数组的实现
2020/12/13 Javascript
部署Python的框架下的web app的详细教程
2015/04/30 Python
Python使用xlrd模块操作Excel数据导入的方法
2015/05/26 Python
python实现根据主机名字获得所有ip地址的方法
2015/06/28 Python
Python语言生成水仙花数代码示例
2017/12/18 Python
Python实现翻转数组功能示例
2018/01/12 Python
在django中实现页面倒数几秒后自动跳转的例子
2019/08/16 Python
Python如何计算语句执行时间
2019/11/22 Python
Python线程障碍对象Barrier原理详解
2019/12/02 Python
Python魔法方法 容器部方法详解
2020/01/02 Python
Python 批量读取文件中指定字符的实现
2020/03/06 Python
Python就将所有的英文单词首字母变成大写
2021/02/12 Python
会计专业毕业生自我鉴定
2013/10/29 职场文书
电台编导求职信
2014/05/06 职场文书