Python中可变和不可变对象的深入讲解


Posted in Python onAugust 02, 2021

前置知识

在 Python 中,一切皆为对象

Python 中不存在值传递,一切传递的都是对象的引用,也可以认为是传址 

有哪些可变对象,哪些不可变对象?

不可变对象:字符串、元组、数字(int、float)

可变对象:数组、字典、集合 

不可变对象和可变对象的区别?

可变对象:改变对象内容,对象在内存中的地址不会被改变

不可变对象:改变对象内容,对象在内存中的地址会被改变;如果必须存储一个不同的值,则必须创建新的对象 

不可变对象的应用场景

它们在需要常量哈希值的地方起着重要作用,例如作为字典中的键

从内存角度出发说下有什么区别?

不可变对象

Python中可变和不可变对象的深入讲解

Python 中的变量有一个内存空间

具体的数据(对象)也有一个内存空间

而变量保存(指向)的是存储数据(对象)的内存地址,一般也叫对象引用

不可变对象是指对象内容本身不可变

变的是:改变了值,会创建新对象,然后变量改变了对象引用,指向了新对象,旧对象会被垃圾回收

可变对象

Python中可变和不可变对象的深入讲解

变的是:原来对象的内容,不会创建新对象,而变量也还是指向原对象

从代码角度看看区别

不可变对象-整型

a = 123
b = a
print(id(a))
print(id(b))
print(a, b)

a += 2

print(id(a))
print(id(b))
print(a, b)


# 输出结果
4473956912
4473956912
123 123
4473956976
4473956912
125 123

从前两次打印可以看到,a、b 变量保存的内存地址是同一个,他们们都保存了 123 的内存地址(123 对象的引用)

预期情况:在 a 做了加法赋值运算之后,既然他们一开始都是指向同一个内存地址,按道理修改 123 后,他们也应该仍然指向同一个内存地址呀,但是并没有!

实际情况:a 指向了新的内存地址,而 b 仍然指向旧的内存地址,所以他们的值也不一样 

可以看看下面的图

首先,这是一个内存区域

Python中可变和不可变对象的深入讲解

原理

因为数字(int、float) 是不可变对象,所以不能在 123 的内存地址上直接修改数据

加法赋值,实际上是将原来的 123 复制了一份到新的内存地址,然后再做加法,得到一个新的值 125,最后 a 再指向新的内存地址

不可变对象-字符串

a = "test"
b = a
print(id(a))
print(id(b))
print(a, b)

a += "123"

print(id(a))
print(id(b))
print(a, b)


# 输出结果
4455345392
4455345392
test test
4455818288
4455345392
test123 test

不可变对象-元组

a = (1, 2, 3)
b = a
print(id(a))
print(id(b))
print(a, b)

a = a + a
print(id(a))
print(id(b))
print(a, b)


# 输出结果
4455410240
4455410240
(1, 2, 3) (1, 2, 3)
4455359200
4455410240
(1, 2, 3, 1, 2, 3) (1, 2, 3)

可变对象列表

# 列表
a = [1, 2, 3]
b = a

print(id(a))
print(id(b))
print(a, b)

a += [4, 5, 6]

print(a, b)
print(id(a))
print(id(b))


# 输出结果
4327665856
4327665856
[1, 2, 3, 4, 5, 6] [1, 2, 3, 4, 5, 6]
4327665856
4327665856

能看到 a 变量修改值之后,b 的值也随之修改了

可以看看下面的图

Python中可变和不可变对象的深入讲解 

因为 list 是不可变对象,所以并不会将原来的值复制到新的内存地址再改变,而是直接在原来的内存地址上修改数据
因为 a、b 都是指向原来的内存地址的,所以 a、b 变量保存的内存地址是一致的(对象引用是一致的),当然值也是一样的啦 

Python 函数的参数传递

这里先提前讲下函数的入门,因为参数传递是个挺重要的点

概念

开头有讲到,Python 的一切传递都是对象的引用,函数参数传递也不例外

当传递给函数的是一个变量,实际上传递的是变量保存的对象引用(变量指向的内存地址)

在函数内部修改变量时,会根据变量指向的内存地址,去修改对应的值才对,事实真是如此吗

参数传递不可变对象

# 函数
def test_no_define(age, name):
    age = 123
    name = "poloyy"
    print(age, name)


age = 1
name = "yy"
print(age, name)

test_no_define(age, name)
print(age, name)


# 输出结果
1 yy
123 poloyy
1 yy

参数传递可变对象

# 函数
def test_define(dicts, sets):
    dicts['age'] = 24
    sets.pop()
    print(dicts, sets)


dicts = {"age": 123}
sets = {1, 2}
print(dicts, sets)

test_define(dicts, sets)
print(dicts, sets)


# 输出结果
1 yy
{'age': 123} {1, 2}
{'age': 24} {2}
{'age': 24} {2}

总结

当函数参数传递的变量是不可变对象的时候,函数内改变变量值,函数外的变量不会随之改变

当函数参数传递的变量是可变对象的时候,函数内改变变量值,函数外的变量会随之改变

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

Python 相关文章推荐
Python的Django框架中的表单处理示例
Jul 17 Python
Python的collections模块中的OrderedDict有序字典
Jul 07 Python
对numpy中数组元素的统一赋值实例
Apr 04 Python
浅谈python中requests模块导入的问题
May 18 Python
python如何获取当前文件夹下所有文件名详解
Jan 25 Python
详解python中init方法和随机数方法
Mar 13 Python
Python+OpenCv制作证件图片生成器的操作方法
Aug 21 Python
利用Python小工具实现3秒钟将视频转换为音频
Oct 29 Python
Python数据可视化:幂律分布实例详解
Dec 07 Python
Python3标准库之threading进程中管理并发操作方法
Mar 30 Python
怎么用Python识别手势数字
Jun 07 Python
Pytest中conftest.py的用法
Jun 27 Python
Python基础数据类型tuple元组的概念与用法
Aug 02 #Python
opencv用VS2013调试时用Image Watch插件查看图片
基于python定位棋子位置及识别棋子颜色
Python 处理表格进行成绩排序的操作代码
python识别围棋定位棋盘位置
python之基数排序的实现
Jul 26 #Python
python之PySide2安装使用及QT Designer UI设计案例教程
You might like
php判断终端是手机还是电脑访问网站的思路及代码
2013/04/24 PHP
PHP数据库操作之基于Mysqli的数据库操作类库
2014/04/19 PHP
[原创]ThinkPHP中SHOW_RUN_TIME不能正常显示运行时间的解决方法
2015/10/10 PHP
PHP简单操作MongoDB的方法(安装及增删改查)
2016/05/26 PHP
网站被恶意镜像怎么办 php一段代码轻松搞定(全面版)
2018/10/23 PHP
用js重建星际争霸
2006/12/22 Javascript
Prototype Template对象 学习
2009/07/19 Javascript
JavaScript高级程序设计(第3版)学习笔记2 js基础语法
2012/10/11 Javascript
利用百度地图JSAPI生成h7n9禽流感分布图实现代码
2013/04/15 Javascript
利用js实现在浏览器状态栏显示访问者在本页停留的时间
2013/12/29 Javascript
类似天猫商品详情随浏览器移动的示例代码
2014/02/27 Javascript
JS动态加载当前时间的方法
2015/02/09 Javascript
jQuery插件scroll实现无缝滚动效果
2015/04/27 Javascript
使用nodejs开发cli项目实例
2015/06/03 NodeJs
HTML5实现留言和回复页面样式
2015/07/22 Javascript
基于javascript实现九九乘法表
2016/03/27 Javascript
浅谈JavaScript变量的自动转换和语句
2016/06/12 Javascript
AngularJS使用带属性值的ng-app指令实现自定义模块自动加载的方法
2017/01/04 Javascript
JavaScript日期库date-fn.js使用方法解析
2020/09/09 Javascript
原生js实现自定义滚动条组件
2021/01/20 Javascript
Python实现微信公众平台自定义菜单实例
2015/03/20 Python
Python3使用requests包抓取并保存网页源码的方法
2016/03/15 Python
Python网络爬虫项目:内容提取器的定义
2016/10/25 Python
Python 数据处理库 pandas进阶教程
2018/04/21 Python
python使用flask与js进行前后台交互的例子
2019/07/19 Python
对Django外键关系的描述
2019/07/26 Python
python处理自动化任务之同时批量修改word里面的内容的方法
2019/08/23 Python
Django生成PDF文档显示网页上以及PDF中文显示乱码的解决方法
2019/12/17 Python
香港个人化生活购物网站:Ballyhoo Limited
2016/09/10 全球购物
好药师网上药店:安全合法的网上药品零售药房
2017/02/15 全球购物
印度在线杂货店:bigbasket
2018/08/23 全球购物
J2ee常用的设计模式?说明工厂模式
2015/05/21 面试题
体育教育毕业生自荐信
2014/06/29 职场文书
我爱祖国演讲稿
2014/09/02 职场文书
研修心得体会
2014/09/04 职场文书
大学生违纪检讨书300字
2014/10/25 职场文书