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中Flask框架简单入门实例
Mar 21 Python
Python中的测试模块unittest和doctest的使用教程
Apr 14 Python
Python3写入文件常用方法实例分析
May 22 Python
基于Python中capitalize()与title()的区别详解
Dec 09 Python
Python基于分析Ajax请求实现抓取今日头条街拍图集功能示例
Jul 19 Python
python下PyGame的下载与安装过程及遇到问题
Aug 04 Python
Python递归函数 二分查找算法实现解析
Aug 12 Python
python用线性回归预测股票价格的实现代码
Sep 04 Python
Python使用matplotlib绘制三维参数曲线操作示例
Sep 10 Python
python实现简单图书管理系统
Nov 22 Python
Django User 模块之 AbstractUser 扩展详解
Mar 11 Python
Python一些基本的图像操作和处理总结
Jun 23 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/01/13 PHP
php对称加密算法示例
2014/05/07 PHP
ThinkPHP3.1数据CURD操作快速入门
2014/06/19 PHP
php使用$_POST或$_SESSION[]向js函数传参
2014/09/16 PHP
PHP错误机制知识汇总
2016/03/24 PHP
thinkphp ajaxfileupload实现异步上传图片的示例
2017/08/28 PHP
PHP+redis实现的限制抢购防止商品超发功能详解
2019/09/19 PHP
utf8的编码算法 转载
2006/12/27 Javascript
Javascript 写的简单进度条控件
2008/01/22 Javascript
jQuery hover 延时器实现代码
2011/03/12 Javascript
JavaScript的继承的封装介绍
2013/10/15 Javascript
鼠标悬浮显示二级菜单效果的jquery实现
2014/10/29 Javascript
jQuery中:gt选择器用法实例
2014/12/29 Javascript
jquery实现的点击翻书效果代码
2015/11/04 Javascript
封装属于自己的JS组件
2016/01/27 Javascript
Node.js开启Https的实践详解
2016/10/25 Javascript
vue-resource 拦截器使用详解
2017/02/21 Javascript
JavaScript中object和Object的区别(详解)
2017/02/27 Javascript
微信小程序中使用ECharts 异步加载数据的方法
2018/06/27 Javascript
解决element ui select下拉框不回显数据问题的解决
2019/02/20 Javascript
Vue双向绑定实现原理与方法详解
2020/05/07 Javascript
仿照Element-ui实现一个简易的$message方法
2020/09/14 Javascript
Linux环境下MySQL-python安装过程分享
2015/02/02 Python
Python性能优化技巧
2015/03/09 Python
Python中的Numeric包和Numarray包使用教程
2015/04/13 Python
Python3实现从指定路径查找文件的方法
2015/05/22 Python
Python扫描IP段查看指定端口是否开放的方法
2015/06/09 Python
Python格式化输出字符串方法小结【%与format】
2018/10/29 Python
python仿evething的文件搜索器实例代码
2019/05/13 Python
Python如何对齐字符串
2020/07/30 Python
英国殿堂级有机护肤品牌:Rodial
2017/04/17 全球购物
基层领导干部“四风”问题批评与自我批评
2014/09/23 职场文书
学生党员批评与自我批评
2014/10/15 职场文书
2015年师德表现自我评价
2015/03/05 职场文书
贷款收入证明格式
2015/06/24 职场文书
Nginx 502 bad gateway错误解决的九种方案及原因
2022/08/14 Servers