一道问题引出的python中可变数据类型与不可变数据类型

1|0一. 问题的提出


我们先来看两个对比

第一道题,当对象为整数时,最终结果:b = 2, a = 1,b的变化没有引起a的变化

a = 1 b = a b += 1 print(a) print(b) 结果: 1 2

第二道题,当对象为字典时,最终结果:a = {"name":"jack","age":27}, b = {"name":"jack","age":27},b的变化引起了a的变化

a = {"name": "Jack"} b = a b["age"] = 27 print(a) print(b) 结果: {"name":"Jack""age":27} {"name":"Jack""age":27}

那么,为什么上述两道题的结果会有如此大的区别呢?我们就来了解一下可变数据类型和不变数据类型

2|0二. 可变数据类型和不可变数据类型


  • 可变数据类型有:列表list、字典dict
  • 不可变数据类型有:整型int、浮点型float、字符串string和元组tuple

python中有一个id()函数,可用来获取对象的内存地址,我们针对第一题,可以看一下a和b指向的对象的内存地址有什么变化

>>> a = 1 >>> b = a >>> id(a) 1814284368 >>> id(b) 1814284368 >>> b += 1 >>> id(b) 1814284400 >>> id(a) 1814284368 >>>

我们可以清晰的看到,a = 1, b = a,这两个操作中,变量a和b指向的对象的内存地址是一样的,也就是说a和b其实引用了同一个对象1。那为什么整型是不可变数据类型呢?这里可以理解为a和b的引用地址处的值是不能被改变的,也就是1814284368地址处的值在没被垃圾回收之前一直是1,不能改变,如果把b赋值为2,那么,只能把b的引用地址从1814284368变为1814284400,相当于b += 1这个赋值又创建了一个新的对象2,然后变量a仍然指向对象1,而变量b指向了对象2,变量b的变化并不会引起a的改变,因为它们指向的是不同的对象

我们大概画个演示图:

图一:a = 1, b = a

 图二: b += 1

我们再来看字典,先用id()看一下内存地址的变化

>>> a = {"name": "Jack"} >>> b = a >>> id(a) 59071624 >>> id(b) 59071624 >>> b["age"] = 27 >>> id(a) 59071624 >>> id(b) 59071624 >>> a {'age': 27, 'name': 'Jack'} >>> b {'age': 27, 'name': 'Jack'}

可以看到,变量a和b同时指向一个字典对象,当给变量b指向的字典添加元素后,b指向的字典的内存地址并不会发生变化,也就是说,对b的操作不会改变a引用的地址值,相当于同样一个地址得到了扩充,由于a和b指向同一个地址,所以b的变化会引起a的变化

图一:a = {"name": "Jack"}, b = a

图二:b["age"] = 27

3|0 三. 参考文章


https://blog.csdn.net/dan15188387481/article/details/49864613


__EOF__

本文作者cnhkzyy
本文链接https://www.cnblogs.com/my_captain/p/9066286.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   cnhkzyy  阅读(418)  评论(2编辑  收藏  举报
编辑推荐:
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
点击右上角即可分享
微信分享提示