python新手学习可变和不可变对象


Posted in Python onJune 11, 2020

python中有可变对象和不可变对象,可变对象:list,dict.不可变对象有:int,string,float,tuple.

python不可变对象

int,string,float,tuple

先来看一个例子

def int_test(): 
 i = 77
 j = 77
 print(id(77))     #140396579590760
 print('i id:' + str(id(i)))  #i id:140396579590760
 print('j id:' + str(id(j)))  #j id:140396579590760
 print i is j     #True
 j = j + 1
 print('new i id:' + str(id(i))) #new i id:140396579590760
 print('new j id:' + str(id(j))) #new j id:140396579590736
 print i is j     #False
 
if __name__ == '__main__':
 int_test()

有i和j俩个变量的值为77,通过打印77的ID和变量i,j在内存中的id我们得知它们都是指向同一块内存。所以说i和j都是指向同一个对象的。然后我们修改j的值,让j的值+1.按道理j修改之后应该i的值也发生改变的,因为它们都是指向的同一块内存,但结果是并没有。因为int类型是不可变类型,所有其实是j复制了一份到新的内存地址然后+1,然后j又指向了新的地址。所以j的内存id发生了变化。

内存分配情况如下:

python新手学习可变和不可变对象

有i和j俩个变量的值为77,通过打印77的ID和变量i,j在内存中的id我们得知它们都是指向同一块内存。所以说i和j都是指向同一个对象的。然后我们修改j的值,让j的值+1.按道理j修改之后应该i的值也发生改变的,因为它们都是指向的同一块内存,但结果是并没有。因为int类型是不可变类型,所有其实是j复制了一份到新的内存地址然后+1,然后j又指向了新的地址。所以j的内存id发生了变化。

内存分配情况如下:

def dict_test():
 a = {}
 b = a
 print(id(a))
 a['a'] = 'hhhh'
 print('id a:' + str(id(a)))
 print('a:' + str(a))
 print('id b:' + str(id(b)))
 print('b:' + str(b))if __name__ == '__main__':
 dict_test()

运行结果如下:

140367329543360
id a:140367329543360
a:{'a': 'hhhh'}
id b:140367329543360
b:{'a': 'hhhh'}

可以看到a最早的内存地址id是140367329543360 然后把a赋值给b其实就是让变量b的也指向a所指向的内存空间。然后我们发现当a发生变化后,b也跟着发生变化了,因为list是可变类型,所以并不会复制一份再改变,而是直接在a所指向的内存空间修改数据,而b也是指向该内存空间的,自然b也就跟着改变了。

内存变化如下:

python新手学习可变和不可变对象

python函数的参数传递

由于python规定参数传递都是传递引用,也就是传递给函数的是原变量实际所指向的内存空间,修改的时候就会根据该引用的指向去修改该内存中的内容,所以按道理说我们在函数内改变了传递过来的参数的值的话,原来外部的变量也应该受到影响。

但是上面我们说到了python中有可变类型和不可变类型,这样的话,当传过来的是可变类型(list,dict)时,我们在函数内部修改就会影响函数外部的变量。而传入的是不可变类型时在函数内部修改改变量并不会影响函数外部的变量,因为修改的时候会先复制一份再修改。下面通过代码证明一下:

def test(a_int, b_list):
 a_int = a_int + 1
 b_list.append('13')
 print('inner a_int:' + str(a_int))
 print('inner b_list:' + str(b_list))
if __name__ == '__main__':
 a_int = 5
 b_list = [10, 11]
 test(a_int, b_list)
 print('outer a_int:' + str(a_int))
 print('outer b_list:' + str(b_list))

运行结果如下:

inner a_int:6
inner b_list:[10, 11, '13']
outer a_int:5
outer b_list:[10, 11, '13']

答案显而易见啦,经过test()方法修改后,传递过来的int类型外部变量没有发生改变,而list这种可变类型则因为test()方法的影响导致内容发生了改变。

总结:

在很多的其他语言中在传递参数的时候允许程序员选择值传递还是引用传递(比如c语言加上*号传递指针就是引用传递,而直接传递变量名就是值传递),而python只允许使用引用传递,但是它加上了可变类型和不可变类型,让我们感觉有点混乱了。听说python只允许引用传递是为方便内存管理,因为python使用的内存回收机制是计数器回收,就是每块内存上有一个计数器,表示当前有多少个对象指向该内存。每当一个变量不再使用时,就让该计数器-1,有新对象指向该内存时就让计数器+1,当计时器为0时,就可以收回这块内存了。

知识点扩展:

Python可变对象与不可变对象原理解析

原理

可变对象:list dict set

不可变对象:tuple string int float bool

1. python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。实际上,这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象的引用,就能修改对象的原始值——相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象的引用,就不能直接修改原始对象——相当于通过“传值'来传递对象。

2. 当人们复制可变对象时,就复制了可变对象的引用,如果改变引用的值,则修改了原始的参数。

3. 为了简化内存管理,Python通过引用计数机制实现自动垃圾回收功能,Python中的每个对象都有一个引用计数,用来计数该对象在不同场所分别被引用了多少次。每当引用一次Python对象,相应的引用计数就增1,每当消毁一次Python对象,则相应的引用就减1,只有当引用计数为零时,才真正从内存中删除Python对象。

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

Python 相关文章推荐
编写Python脚本批量下载DesktopNexus壁纸的教程
May 06 Python
基于python爬虫数据处理(详解)
Jun 10 Python
用Python写王者荣耀刷金币脚本
Dec 21 Python
Python浅复制中对象生存周期实例分析
Apr 02 Python
Python使用Selenium模块实现模拟浏览器抓取淘宝商品美食信息功能示例
Jul 18 Python
python实现图片筛选程序
Oct 24 Python
python获取url的返回信息方法
Dec 17 Python
2019 Python最新面试题及答案16道题
Apr 11 Python
django settings.py 配置文件及介绍
Jul 15 Python
pytorch 图像预处理之减去均值,除以方差的实例
Jan 02 Python
解决Python import docx出错DLL load failed的问题
Feb 13 Python
Python包管理工具pip的15 个使用小技巧
May 17 Python
基于Keras 循环训练模型跑数据时内存泄漏的解决方式
Jun 11 #Python
什么是python的id函数
Jun 11 #Python
Keras:Unet网络实现多类语义分割方式
Jun 11 #Python
Pycharm中配置远程Docker运行环境的教程图解
Jun 11 #Python
Keras 快速解决OOM超内存的问题
Jun 11 #Python
python3.6.8 + pycharm + PyQt5 环境搭建的图文教程
Jun 11 #Python
使用keras实现孪生网络中的权值共享教程
Jun 11 #Python
You might like
PHP rsa加密解密使用方法
2015/04/27 PHP
CodeIgniter辅助之第三方类库third_party用法分析
2016/01/20 PHP
php使用yield对性能提升的测试实例分析
2019/09/19 PHP
Javascript实例教程(19) 使用HoTMetal(2)
2006/12/23 Javascript
JS实现的全排列组合算法示例
2017/10/09 Javascript
vue远程加载sfc组件思路详解
2019/12/25 Javascript
[02:32]DOTA2英雄基础教程 美杜莎
2014/01/07 DOTA
Python创建系统目录的方法
2015/03/11 Python
python用来获得图片exif信息的库实例分析
2015/03/16 Python
Python爬虫框架Scrapy实战之批量抓取招聘信息
2015/08/07 Python
python与php实现分割文件代码
2017/03/06 Python
python数据预处理之将类别数据转换为数值的方法
2017/07/05 Python
深入探究Django中的Session与Cookie
2017/07/30 Python
python 3.0 模拟用户登录功能并实现三次错误锁定
2017/11/01 Python
Python实现ping指定IP的示例
2018/06/04 Python
如何利用Python分析出微信朋友男女统计图
2019/01/25 Python
获取django框架orm query执行的sql语句实现方法分析
2019/06/20 Python
Jacobi迭代算法的Python实现详解
2019/06/29 Python
pytorch中的上采样以及各种反操作,求逆操作详解
2020/01/03 Python
python函数enumerate,operator和Counter使用技巧实例小结
2020/02/22 Python
python ImageDraw类实现几何图形的绘制与文字的绘制
2020/02/26 Python
BeautifulSoup中find和find_all的使用详解
2020/12/07 Python
Python字符串对齐、删除字符串不需要的内容以及格式化打印字符
2021/01/23 Python
移动端HTML5 input常见问题(小结)
2020/09/28 HTML / CSS
马来西亚与新加坡长途巴士售票网站:BusOnlineTicket.com
2018/11/05 全球购物
描述内存分配方式以及它们的区别
2016/10/15 面试题
商务英语大学生职业生涯规划书范文
2014/01/01 职场文书
煤矿安全生产责任书
2014/04/15 职场文书
领导干部作风整顿剖析材料
2014/10/11 职场文书
2015年元旦晚会活动总结(学生会)
2014/11/28 职场文书
爱国影片观后感
2015/06/18 职场文书
2016年六一文艺汇演开幕词
2016/03/04 职场文书
导游词之西递宏村
2019/12/10 职场文书
《分一些蚊子进来》读后感3篇
2020/01/09 职场文书
node.js如何自定义实现一个EventEmitter
2021/07/16 Javascript
《进击的巨人》新联动CM 兵长强势出击兽巨人
2022/04/05 日漫