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获取糗百图片代码实例
Dec 18 Python
python实现一次创建多级目录的方法
May 15 Python
Python入门教程之运算符与控制流
Aug 17 Python
python 读取txt中每行数据,并且保存到excel中的实例
Apr 29 Python
Flask框架URL管理操作示例【基于@app.route】
Jul 23 Python
python 实现批量xls文件转csv文件的方法
Oct 23 Python
Python从数据库读取大量数据批量写入文件的方法
Dec 10 Python
pytorch:torch.mm()和torch.matmul()的使用
Dec 27 Python
tensorflow2.0与tensorflow1.0的性能区别介绍
Feb 07 Python
利于python脚本编写可视化nmap和masscan的方法
Dec 29 Python
python开发实时可视化仪表盘的示例
May 07 Python
Python3 如何开启自带http服务
May 18 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中根据IP地址判断城市实现城市切换或跳转代码
2012/09/04 PHP
在PHP模板引擎smarty生成随机数的方法和math函数详解
2014/04/24 PHP
tp5.0框架隐藏index.php入口文件及模块和控制器的方法分析
2020/02/11 PHP
WordPress 插件——CoolCode使用方法与下载
2007/07/02 Javascript
FileUpload上传图片(图片不变形)
2010/08/05 Javascript
Jquery的hide及toggle方法让超链接慢慢消失
2013/09/06 Javascript
javascript中不等于的代码是什么怎么写
2013/12/29 Javascript
button没写type=button会导致点击时提交
2014/03/06 Javascript
JavaScript制作windows经典扫雷小游戏
2015/03/31 Javascript
jQuery实现的左右移动焦点图效果
2016/01/14 Javascript
JS组件Bootstrap Select2使用方法详解
2020/04/17 Javascript
js+css实现select的美化效果
2016/03/24 Javascript
jquery ajax局部加载方法详解(实现代码)
2016/05/12 Javascript
JavaScript提升性能的常用技巧总结【经典】
2016/06/20 Javascript
AngularJS中的包含详细介绍及实现示例
2016/07/28 Javascript
AngularJs Understanding the Model Component
2016/09/02 Javascript
javascript中href和replace的比较(详解)
2016/11/25 Javascript
详解node中创建服务进程
2017/05/09 Javascript
Nodejs中获取当前函数被调用的行数及文件名详解
2018/12/12 NodeJs
Typescript3.9 常用新特性一览(推荐)
2020/05/14 Javascript
[00:31]DOTA2上海特级锦标赛 Fnatic战队宣传片
2016/03/04 DOTA
Python 过滤字符串的技巧,map与itertools.imap
2008/09/06 Python
python 获取文件列表(或是目录例表)
2009/03/25 Python
Python socket网络编程TCP/IP服务器与客户端通信
2017/01/05 Python
解决python3捕获cx_oracle抛出的异常错误问题
2018/10/18 Python
利用Python查看微信共同好友功能的实现代码
2019/04/24 Python
python机器学习库scikit-learn:SVR的基本应用
2019/06/26 Python
Python Web版语音合成实例详解
2019/07/16 Python
使用python实现数组、链表、队列、栈的方法
2019/12/20 Python
Python3 mmap内存映射文件示例解析
2020/03/23 Python
Python Tornado实现WEB服务器Socket服务器共存并实现交互的方法
2020/05/26 Python
手把手教你如何用Pycharm2020.1.1配置远程连接的详细步骤
2020/08/07 Python
外企C语言笔试题
2013/11/10 面试题
启动仪式策划方案
2014/06/14 职场文书
学校工作推荐信范文
2014/07/11 职场文书
优秀团员个人总结
2015/02/26 职场文书