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语言中的按位运算符
Nov 26 Python
Python中模块与包有相同名字的处理方法
May 05 Python
python实现单线程多任务非阻塞TCP服务端
Jun 13 Python
Python抓取框架Scrapy爬虫入门:页面提取
Dec 01 Python
python爬虫使用cookie登录详解
Dec 27 Python
解决tensorflow1.x版本加载saver.restore目录报错的问题
Jul 26 Python
Python爬虫库BeautifulSoup获取对象(标签)名,属性,内容,注释
Jan 25 Python
Python3操作MongoDB增册改查等方法详解
Feb 10 Python
python+opencv边缘提取与各函数参数解析
Mar 09 Python
3分钟看懂Python后端必须知道的Django的信号机制
Jul 26 Python
Python selenium爬取微信公众号文章代码详解
Aug 12 Python
python获取天气接口给指定微信好友发天气预报
Dec 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 读取和编写 XML
2014/11/19 PHP
php在线解压ZIP文件的方法
2014/12/30 PHP
php类常量用法实例分析
2015/07/09 PHP
Thinkphp5+plupload实现的图片上传功能示例【支持实时预览】
2019/05/08 PHP
JScript的条件编译
2007/05/29 Javascript
基于Jquery的表格隔行换色,移动换色,点击换色插件
2010/12/22 Javascript
扩展IE中一些不兼容的方法如contains、startWith等等
2014/01/09 Javascript
解决json日期格式问题的3种方法
2014/02/02 Javascript
动态读取JSON解析键值对的方法
2014/06/03 Javascript
js实现简单选项卡与自动切换效果的方法
2015/04/10 Javascript
jQuery中通过ajax的get()函数读取页面的方法
2016/02/29 Javascript
对jQuary选择器的全面总结
2016/06/20 Javascript
JavaScript探测CSS动画是否已经完成的方法
2016/08/30 Javascript
AngularJs  Using $location详解及示例代码
2016/09/02 Javascript
JS实现的幻灯片切换显示效果
2016/09/07 Javascript
Vue仿手机qq的实例代码(demo)
2017/09/08 Javascript
JS严格模式知识点总结
2018/02/27 Javascript
ES6顶层对象、global对象实例分析
2019/06/14 Javascript
vue之组件内监控$store中定义变量的变化详解
2019/11/08 Javascript
python调用机器喇叭发出蜂鸣声(Beep)的方法
2015/03/23 Python
django实现分页的方法
2015/05/26 Python
Python中的集合类型知识讲解
2015/08/19 Python
详解python之多进程和进程池(Processing库)
2017/06/09 Python
Python基于百度云文字识别API
2018/12/13 Python
Python使用pyserial进行串口通信的实例
2019/07/02 Python
Python  Django 母版和继承解析
2019/08/09 Python
python 内置函数汇总详解
2019/09/16 Python
Python爬虫库requests获取响应内容、响应状态码、响应头
2020/01/25 Python
详解pandas映射与数据转换
2021/01/22 Python
介绍一些UNIX常用简单命令
2014/11/11 面试题
劳资协议书范本
2014/04/23 职场文书
祖国在我心中演讲稿450字
2014/09/05 职场文书
民事调解书范文
2015/05/20 职场文书
房贷收入证明范本
2015/06/12 职场文书
大学学习委员竞选稿
2015/11/20 职场文书
交通安全宣传标语(100条)
2019/08/22 职场文书