Python可变与不可变数据和深拷贝与浅拷贝


Posted in Python onApril 06, 2022

浅拷贝和深拷贝

拷贝函数是专门为可变数据类型listsetdict使用的一种函数。作用是,当一个值指向另一个值的时候,也不会影响指向的值,如果被指向的数据是可变数据,那么它一旦被修改,指向的数据也会随之改变。

什么是可变数据和不可变数据

我们来举一个例子,整型是不可变的数据,那么为什么是不可变的数据呢?一个数据是不是可变的就要关系到python的缓存机制。

当一个数据发生变化,如果它的内存地址没有发生变化,就说明这是一个可变数据。

比如说,我们现在创建一个值是a的变量,它的值是100,然后让这个数值发生变化,观察者个变量的内存地址是否发生了变化。

a = 100
print(a, id(a)) # 100 1610845392

a += 100
print(a, id(a)) # 200 1610848592

我们发现数值发生了变化,变量的内存也跟着发生了变化,我们再创建一个变量b,值也是整型100

b = 100
print(b, id(b))	# 100 1610845392

发现b的内存地址和a的内存地址是一样的,也就是说,像整型这样的数据类型,一个数字就独占一个内存地址,当某个指向这个值的变量,发生了变化的时候,不是这个变量的值要改变,而是这个变量要寻找改变后的值的内存地址,然后重新的指向它。只要你的硬件不重新启动,那么这个内存地址就永远也不会发生变化了,这样的数据就是不可变数据。

那么,反之就是可变数据,指的就是当变量指向的值发生变化之后,在这个内存地址上的值实打实的发生变化的值,就是可变数据类型。

比如列表,列表发生改变之后,是在原有的基础上发生变化的,所以内存地址是不会改变的,这就是可变数据类型,可变数据类型没有内存缓存机制,不能节省内存,所以一模一样的数据,他们的内存地址可能是不相同的。

a = [1, 2]
print(a, id(a)) # [1, 2] 1528536069896

a.append(3)
print(a, id(a)) # [1, 2, 3] 1528536069896

# b 和 a的值相同,但是内存地址不相同
b = [1, 2, 3]
print(b, id(b)) # [1, 2, 3] 1528536069832

那么拷贝函数是干什么的?

在我们的实际工作当中,经常会使用的一种操作就是定义一个变量,它的值直接就赋给了一个原有的变量之上。可是变量定义之后我们绝不是用来作为一个摆设的,而是要做运算、或者是做一个临时的存储,那么原有的变量的值是要改变的,问题就来了,如果是一个不可变的数据还好,如果是可变的数据,直接的赋值他们的内存地址是相同的, 如果一个变量的值发生变化,同内存地址的的值就都发生改变了,我们的向要临时存储的值也就不再是我们想要的那个值了,这是绝大多数的时候我们不想看到的结果。

我们拿整型为例,变量a直接赋值给变量b,这个时候的变量a b 的值是相同的,但是如果变量a的值发生了变化,是丝毫不影响变量b的值的。

a = 100
print(a, id(a))  # 100 1610845392

b = a
print(b, id(b))  # 100 1610845392

a += 100
print(a, id(a))  # 200 1610848592
print(b, id(b))  # 100 1610845392

但是如果是可变数据就不是这样的情况了

a = [1, 2]
print(a, id(a))  # [1, 2] 2077688035080

b = a
print(b, id(b))  # [1, 2] 2077688035080

a.append(3)
print(a, id(a))  # [1, 2, 3] 2077688035080
print(b, id(b))  # [1, 2, 3] 2077688035080

不可变数据的这个特性既是一个优点也是一个缺点,缺点就是如果我们想要保存a变量发生变化之前的的一个状况的时候,是保存不下来的,这个时候就出现了拷贝函数,它可以将可变数据变成不可变数据那样的效果。

浅拷贝

使用拷贝函数,将a变量放入作为参数放入函数中,使用b变量接受函数的返回值,就成功的拷贝了变量a,变量b的内存地址和变量a的不一样,这样当它们其中一方发生变化之后,不会影响到另一方的数据。

# 拷贝函数不能直接使用,需要使用import导入copy模块,copy模块的copy函数就是浅拷贝

import copy

a = [1, 2, 3]

# 变量b不在直接是变量a的直接赋值了,而是通过copy函数的返回值
b = copy.copy(a)

# 他们的数值一样,但是内存地址不同,所以他们之间的任意一方发生变化都不会影响到第二方。
print(a, id(a))  # [1, 2, 3] 2343743813320
print(b, id(b))  # [1, 2, 3] 2343743813192

a.append(4)
print(a, id(a))  # [1, 2, 3, 4] 2343743813320
print(b, id(b))  # [1, 2, 3] 2343743813192

但是如果变量a是一个二级容器或者是一个更多级容器,浅拷贝无法拷贝第二级容器或者更多级的容器,所以当第二级容器或者是更多级的容器发生变化的时候,还是会发生变化,因为浅拷贝只能拷贝一级容器,所以多级容器的内存地址还是相同的。

import copy

a = [[66,88], 2, 3]

b = copy.copy(a)

print(a, id(a))  # [[66, 88], 2, 3] 2431683163720
print(b, id(b))  # [[66, 88], 2, 3] 2431683162184

# 改变二级容器
a[0].append(100)
print(a, id(a))  # [[66, 88, 100], 2, 3] 2431683163720
print(b, id(b))  # [[66, 88, 100], 2, 3] 2431683162184

# 浅拷贝不能拷贝二级及以上的容器
print(id(a[0]))  # 1582481372872
print(id(b[0]))  # 1582481372872

深拷贝

浅拷贝只能拷贝一级容器

所以诞生了深拷贝,深拷贝可以拷贝所有级别的容器。

import copy

a = [[66,88], 2, 3]

# 深拷贝使用deepcopy函数
b = copy.deepcopy(a)


print(a, id(a))  # [[66, 88], 2, 3] 2168411158088
print(b, id(b))  # [[66, 88], 2, 3] 2168411156552

a[0].append(100)
print(a, id(a))  # [[66, 88, 100], 2, 3] 2168411158088
print(b, id(b))  # [[66, 88], 2, 3] 2168411156552

# 深拷贝所有级别的容器
print(id(a[0]))  # 2168411158216
print(id(b[0]))  # 2168411122760

总结

使用深浅拷贝需要导入copy模块;

浅拷贝使用copy函数,只能拷贝一级容器的所有元素;

深拷贝使用deepcopy函数,可以拷贝所有级别容器的所有元素;

标准库copy中只有copydeepcopy两个函数对外开放使用;

因为深拷贝要拷贝的元素跟多,所以速度会远不如浅拷贝,在编程的过程中要注意避免造成多余的系统负担;

python中的不可变数据是Number、string、tuple,可变数据是list、set、dict;而拷贝就是专门为可变数据提供的,所以深浅拷贝只适用于list、set、dict,当然,可变数据使用拷贝函数也不会出错,但是没有意义。

到此这篇关于Python可变与不可变数据和深拷贝与浅拷贝的文章就介绍到这了,更多相关Python数据与拷贝内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
Python中tell()方法的使用详解
May 24 Python
Python中import导入上一级目录模块及循环import问题的解决
Jun 04 Python
pandas pivot_table() 按日期分多列数据的方法
Nov 16 Python
python实现维吉尼亚加密法
Mar 20 Python
Python分析彩票记录并预测中奖号码过程详解
Jul 09 Python
python 求定积分和不定积分示例
Nov 20 Python
python定间隔取点(np.linspace)的实现
Nov 27 Python
pyecharts动态轨迹图的实现示例
Apr 17 Python
python爬虫实现爬取同一个网站的多页数据的实例讲解
Jan 18 Python
Python实现简繁体转换
Jun 07 Python
python 网络编程要点总结
Jun 18 Python
PYTHON基于Pyecharts绘制常见的直角坐标系图表
Apr 28 Python
Python 全局空间和局部空间
Apr 06 #Python
Selenium浏览器自动化如何上传文件
Apr 06 #Python
在Python 中将类对象序列化为JSON
Apr 06 #Python
Python中itertools库的四个函数介绍
Apr 06 #Python
如何用六步教会你使用python爬虫爬取数据
基于Python实现射击小游戏的制作
python使用opencv对图像添加噪声(高斯/椒盐/泊松/斑点)
You might like
常用的php对象类型判断
2008/08/27 PHP
php实现TCP端口检测的方法
2015/04/01 PHP
基于JavaScript自定义构造函数的详解说明
2013/04/24 Javascript
5种处理js跨域问题方法汇总
2014/12/04 Javascript
jQuery DOM删除节点操作指南
2015/03/03 Javascript
jquery实现超简洁的TAB选项卡效果代码
2015/08/28 Javascript
省市联动效果的简单实现代码(推荐)
2016/06/06 Javascript
JSP基于Bootstrap分页显示实例解析
2016/06/12 Javascript
vue 怎么创建组件及组件使用方法
2017/07/27 Javascript
arcgis for js栅格图层叠加(Raster Layer)问题
2017/11/22 Javascript
原生JS封装_new函数实现new关键字的功能
2018/08/12 Javascript
js实现点击展开隐藏效果(实例代码)
2018/09/28 Javascript
微信小程序MUI侧滑导航菜单示例(Popup弹出式,左侧不动,右侧滑动)
2019/01/23 Javascript
利用JS响应式修改vue实现页面的input值
2019/09/02 Javascript
layui使用label标签的方法
2019/09/14 Javascript
微信小程序pinker组件使用实现自动相减日期
2020/05/07 Javascript
微信小程序自定义顶部组件customHeader的示例代码
2020/06/03 Javascript
[36:22]VP vs Serenity 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
[03:01]完美世界DOTA2联赛PWL S2 集锦第二期
2020/12/03 DOTA
使用Python获取Linux系统的各种信息
2014/07/10 Python
Python使用cx_Oracle调用Oracle存储过程的方法示例
2017/10/07 Python
selenium python浏览器多窗口处理代码示例
2018/01/15 Python
python3 实现验证码图片切割的方法
2018/12/07 Python
python 的 scapy库,实现网卡收发包的例子
2019/07/23 Python
Django学习之文件上传与下载
2019/10/06 Python
Python实现分数序列求和
2020/02/25 Python
女性时尚在线:IVRose
2019/02/23 全球购物
腾讯技术类校园招聘笔试试题
2014/05/06 面试题
大学生关于奋斗的演讲稿
2014/01/09 职场文书
工作决心书
2014/03/11 职场文书
教师四风问题对照检查材料
2014/09/26 职场文书
法定代表人授权委托书范本
2014/10/07 职场文书
安全先进班组材料
2014/12/26 职场文书
赵氏孤儿观后感
2015/06/09 职场文书
《詹天佑》教学反思
2016/02/20 职场文书
my.ini优化mysql数据库性能的十个参数(推荐)
2021/05/26 MySQL