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写的创建文件夹自定义函数mkdir()
Aug 25 Python
python监控网站运行异常并发送邮件的方法
Mar 13 Python
Python基于回溯法子集树模板解决全排列问题示例
Sep 07 Python
Python随机函数random()使用方法小结
Apr 29 Python
python+selenium实现简历自动刷新的示例代码
May 20 Python
Django利用cookie保存用户登录信息的简单实现方法
May 27 Python
PyQt编程之如何在屏幕中央显示窗体的实例
Jun 18 Python
使用Python画出小人发射爱心的代码
Nov 23 Python
简单了解python元组tuple相关原理
Dec 02 Python
基于pytorch的lstm参数使用详解
Jan 14 Python
详解Python 3.10 中的新功能和变化
Apr 28 Python
Python自动操作神器PyAutoGUI的使用教程
Jun 16 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中集成PayPal标准支付的实现方法分享
2012/02/06 PHP
CodeIgniter中使用cookie的三种方式详解
2014/07/18 PHP
利用PHP生成CSV文件简单示例
2016/12/21 PHP
YII框架批量插入数据的方法
2017/03/18 PHP
Swoole扩展的6种模式深入详解
2021/03/04 PHP
Apply an AutoFormat to an Excel Spreadsheet
2007/06/12 Javascript
基于jquery的不规则矩形的排列实现代码
2012/04/16 Javascript
JS网页图片按比例自适应缩放实现方法
2014/01/15 Javascript
javascript作用域链(Scope Chain)用法实例解析
2015/11/30 Javascript
浅谈JavaScript前端开发的MVC结构与MVVM结构
2016/06/03 Javascript
详解为Angular.js内置$http服务添加拦截器的方法
2016/12/20 Javascript
bootstrap输入框组件使用方法详解
2017/01/19 Javascript
详解webpack分包及异步加载套路
2017/06/29 Javascript
JS实现数组简单去重及数组根据对象中的元素去重操作示例
2018/01/05 Javascript
jQuery代码优化方法总结
2018/01/29 jQuery
在vue中v-bind使用三目运算符绑定class的实例
2018/09/29 Javascript
跟老齐学Python之玩转字符串(1)
2014/09/14 Python
利用一个简单的例子窥探CPython内核的运行机制
2015/03/30 Python
解读Python中degrees()方法的使用
2015/05/18 Python
Python selenium 三种等待方式详解(必会)
2016/09/15 Python
Python 删除整个文本中的空格,并实现按行显示
2018/07/24 Python
python 监测内存和cpu的使用率实例
2019/11/28 Python
使用python matploblib库绘制准确率,损失率折线图
2020/06/16 Python
python实现取余操作的简单实例
2020/08/16 Python
如何Tkinter模块编写Python图形界面
2020/10/14 Python
2013年大学生的自我鉴定
2013/10/24 职场文书
母亲80寿诞答谢词
2014/01/16 职场文书
入党积极分子自我鉴定范文
2014/03/25 职场文书
公证书样本
2014/04/10 职场文书
学生保证书范文
2014/04/28 职场文书
篮球兴趣小组活动总结
2014/07/07 职场文书
节水倡议书
2015/01/19 职场文书
工厂采购员岗位职责
2015/04/07 职场文书
离婚起诉书怎么写
2015/05/19 职场文书
靠谱的活动总结
2019/04/16 职场文书
JavaScript严格模式不支持八进制的问题讲解
2021/11/07 Javascript