详解Python直接赋值,深拷贝和浅拷贝


Posted in Python onJuly 09, 2020

直接赋值: 对象的引用,也就是给对象起别名
浅拷贝: 拷贝父对象,但是不会拷贝对象的内部的子对象。
深拷贝: 拷贝父对象. 以及其内部的子对象

在之前的文章中,提到可变对象和不可变对象,接下来也是以这两者的区别进行展开

直接赋值

对于可变对象和不可变对象,将一个变量直接赋值给另外一个变量,两者 id 值一致,其实本质上是将变量量绑定到对象的过程.

>>> a=1
>>> b=a
>>> id(a) == id(b)
True
>>> c="string"
>>> d=c
>>> id(c) == id(d)
True
>>> e=[1,2,3]
>>> f=e
>>> id(e)==id(f)
True

关于修改新变量的值,对原有变量会产生的影响,在可变对象和不可变对象 中也做了讲述,这里通过几个例子,重新温习一下

不可变对象

>>> x=1
>>> y=x
>>> id(x)==id(y)
True
>>> id(1)==id(y)
True
>>>>>> id(x)
1500143776
>>> y=y+1
>>> y
2
>>> x
1
>>> id(x)==id(y)
False
>>> id(y)
1500143808
>>> id(x)
1500143776

对于不可变对象,修改赋值后的新变量,不会对原有变量造成任何影响.为什么出现这种现象呢?因为不可变对象一旦创建之后就不允许被改变.后面对 y 进行的操作,其实是重新创建一个对象并绑定的结果:

详解Python直接赋值,深拷贝和浅拷贝

可变对象

>>> m=[1,2,3]
>>> n=m
>>> id(n)==id(m)
True
>>> id(m)
1772066764488
>>> id(n[0])
1772066764656
>>> n[0]=4
>>> n
[4, 2, 3]
>>> m
[4, 2, 3]
>>> id(n)==id(m)
True
>>> id(m)
1772066764488

对于可变对象,修改赋值后的变量,会对原有的变量造成影响,会导致其 value 值的改变,但是其id 值保持不变

详解Python直接赋值,深拷贝和浅拷贝

从上图不难看出,这个时候的 id(n[0]) 的值,和未修改前的 id值应该不一样,可以输出看一下

>>>id(n[0])
1772066764752 # 最初没有修改前是 1772066764656

n[0] 修改前后为什么 id 值出现改变呢? 首先需要明确一点 n[0] 绑定的是一个不可变对象,在文章的最初提到,不可变对象一旦创建就不允许修改.显然对 n[0] 进行修改,不能在绑定对象的内存上进行修改,那如何实现重新赋值呢?只能创建一个新的对象 4 ,然后将 n[0] 绑定到新的对象

浅拷贝和深拷贝

先看一下官方文档的定义

The difference between shallow and deep copying is only relevant for compound objects (objects that contain other objects, like lists or
class instances).
A shallow copy constructs a new compound object and then (to the
extent possible) inserts the same objects into it that the
original contains.
A deep copy constructs a new compound object and then, recursively,inserts copies into it of the objects found in the original.

从文档中不难看出,上面提到深拷贝和浅拷贝两者区别在于在复合对象,那接下来也只讨论复合对象.

浅拷贝

注意到官方文档也提到对浅拷贝和深拷贝的定义,从上文中不难看出,浅拷贝构建一个复合对象,然后将原有复合对象包含的对象插入到新的复合对象中

详解Python直接赋值,深拷贝和浅拷贝

从上图不难看出,浅拷贝后,新复合对象包含的对象(可变或者不可变)的 id 值和原有对象包含的对象的 id 值相同

看一下具体例子:

>>> import copy
>>> a=[1,2,[3,4]]
>>> b=copy.copy(a)
>>> id(b[0])==id(a[0])
True
>>> id(b[2])==id(a[2])
True
>>> id(b[2][0])==id(a[2][0])
True

现在让我们试着修改一下浅拷贝后的 b 的值,在修改前,可以先思考一下,如果修改 b[0] 可能会发生什么?

由于 b[0] = 1,很显然 1 属于不可变对象,那么根据对不可变变量修改的规则,则 b[0] 会绑定到新的变量上,而 a[0] 的由于没有修改,则保持不变,真的是这样吗?让我们验证一下

>>> b[0]=5
>>> b
[5, 2, [3, 4]]
>>> a
[1, 2, [3, 4]]

接下来我们要尝试修改一下 b[2],由于 b[2] 绑定的对象是 list,属于可变对象,按照上面说的可变对象修改的规则,则修改后的 b[2]id 值保持不变,但是其 value 值会发生改变. 同样的让我们通过例子验证一下

>>> id(b[2])
4300618568
>>> b[2][0]=6
>>> id(b[2])
4300618568
>>> b
[5, 2, [6, 4]]
>>> a
[1, 2, [6, 4]]

由于 b[2]a[2] 绑定同一个可变对象,很显然对 b[2] 的修改同样会映射到 a[2]

深拷贝

深拷贝构建一个复合对象,然后递归的将原有复合包含的对象的副本插入到新的复合对象中

详解Python直接赋值,深拷贝和浅拷贝

若上图所示,深拷贝后,新的复合对象包含的对象,若对象为不可变对象,则 id 值保持不变,若对象为可变对象,则 id 值发生改变

看一个例子:

>>> import copy
>>> a=[1,2,[3,4]]
>>> b=copy.deepcopy(a)
>>> id(b[0])==id(a[0])
True
>>> id(b[2])==id(a[0])
False
>>> id(b[2][0])==id(a[2][0])
True

接下来让我们修改一下变量 b,这里就不在修改不可变对象 b[0]b[1] 了,因为结果很明显,对 a 不会产生任何影响,我们来修改 b[2],那么修改 b[2] 会对 a[2] 产生影响吗?很明显答案是不会,因为深拷贝就相当于克隆出了一个全新的个体,两者不再有任何关系

>>> b[2][0]=5
>>> b
[1, 2, [5, 4]]
>>> a
[1, 2, [3, 4]]

以上就是详解Python直接赋值,深拷贝和浅拷贝的详细内容,更多关于Python直接赋值,深拷贝和浅拷贝的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
使用Python的web.py框架实现类似Django的ORM查询的教程
May 02 Python
python记录程序运行时间的三种方法
Jul 14 Python
Python基于time模块求程序运行时间的方法
Sep 18 Python
python多线程并发让两个LED同时亮的方法
Feb 18 Python
python3 打印输出字典中特定的某个key的方法示例
Jul 06 Python
Python BeautifulSoup [解决方法] TypeError: list indices must be integers or slices, not str
Aug 07 Python
用python3读取python2的pickle数据方式
Dec 25 Python
pycharm 中mark directory as exclude的用法详解
Feb 14 Python
浅谈Python 钉钉报警必备知识系统讲解
Aug 17 Python
Python web框架(django,flask)实现mysql数据库读写分离的示例
Nov 18 Python
python中使用asyncio实现异步IO实例分析
Feb 26 Python
Python爬虫中urllib3与urllib的区别是什么
Jul 21 Python
使用python实现下载我们想听的歌曲,速度超快
Jul 09 #Python
Python selenium模块实现定位过程解析
Jul 09 #Python
Python自动化操作实现图例绘制
Jul 09 #Python
解决pip install psycopg2出错问题
Jul 09 #Python
利用python下载scihub成文献为PDF操作
Jul 09 #Python
Python scrapy爬取小说代码案例详解
Jul 09 #Python
使用Python实现微信拍一拍功能的思路代码
Jul 09 #Python
You might like
php通过修改header强制图片下载的方法
2015/03/24 PHP
CodeIgniter扩展核心类实例详解
2016/01/20 PHP
PHP中引用类型和值类型功能与用法示例
2019/02/26 PHP
laravel框架与其他框架的详细对比
2019/10/23 PHP
jQuery Ajax 全解析
2009/02/08 Javascript
简短几句jquery代码的实现一个图片向上滚动切换
2011/09/02 Javascript
jquery移动listbox的值原理及代码
2013/05/03 Javascript
自动刷新网页,自动刷新当前页面,JS调用
2013/06/24 Javascript
jQuery中DOM树操作之复制元素的方法
2015/01/23 Javascript
javascript中Array数组的迭代方法实例分析
2015/02/04 Javascript
javascript实现捕捉键盘上按下的键
2015/05/05 Javascript
设计模式中的组合模式在JavaScript程序构建中的使用
2016/05/18 Javascript
JS生成和下载二维码的代码
2016/12/07 Javascript
vue 2.0组件与v-model详解
2017/03/27 Javascript
vue.js国际化 vue-i18n插件的使用详解
2017/07/07 Javascript
基于Vue2的独立构建与运行时构建的差别(详解)
2017/12/06 Javascript
[01:23:59]2018DOTA2亚洲邀请赛 4.1 小组赛 B组 VP vs Secret
2018/04/03 DOTA
python正则表达式抓取成语网站
2013/11/20 Python
python复制与引用用法分析
2015/04/08 Python
详谈Python2.6和Python3.0中对除法操作的异同
2017/04/28 Python
python实现合并两个排序的链表
2019/03/03 Python
PyQt5固定窗口大小的方法
2019/06/18 Python
numpy:找到指定元素的索引示例
2019/11/26 Python
Python合并2个字典成1个新字典的方法(9种)
2019/12/19 Python
解决paramiko执行命令超时的问题
2020/04/16 Python
沪江旗下的海量优质课程平台:沪江网校
2017/11/07 全球购物
幼儿园教师培训制度
2014/01/16 职场文书
石油大学毕业生自荐信
2014/01/28 职场文书
小学毕业感言150字
2014/02/05 职场文书
农业局学习党的群众路线教育实践活动心得体会
2014/03/07 职场文书
2014年收银工作总结
2014/11/13 职场文书
2015年感恩节演讲稿(优选篇)
2015/03/20 职场文书
小学作文指导之如何写人?
2019/07/08 职场文书
sql查询语句之平均分、最高最低分及排序语句
2022/05/30 MySQL
MySQL外键约束(Foreign Key)案例详解
2022/06/28 MySQL
JS高级程序设计之class继承重点详解
2022/07/07 Javascript