内存中的Python:Python引用计数指南
云栖号资讯:【点击查看更多行业资讯】
在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来!
本文将会为你介绍Python引用计数,演示中使用可变列表对象,不过本文不介绍C语言实现细节。
需要注意的是,代码段的输出在硬件上可能有所不同。
变量是内存引用
Python中的变量是内存引用。如果输入x = [1,2]时会发生什么?[1,2]是对象。
回想一下,一切都是Python中的对象。[1,2]将在内存中创建。x是[1,2]对象的内存引用。
来看看下面的例子。可以找到x所引用的内存地址。请务必只使用id(x),它会以10为基数,而十六进制函数会将其转换为十六进制。
x = [1, 2] print(hex(id(x))) # output: 0x32ebea8
引用计数
现在已经在内存中创建了一个list对象,而且x对该对象进行了引用。那么y=[1,2]和y=x有什么区别?
当输入y=[1,2]时,它将在内存中创建一个新的list对象,并且y将引用它。
x = [1, 2] y = [1, 2] print(hex(id(x))) # output: 0x101bea8 print(hex(id(y))) # output: 0x31a5528
而当输入y=x时,等同于告诉Python希望y 变量引用x变量引用的内容。因为变量是内存引用的。
可以确认x和y引用同一个对象。
x = [1, 2] y = x print(hex(id(x))) # output: 0x74bea8 print(hex(id(y))) # output: 0x74bea8
引用计数的数目
接下来的问题是,有多少变量引用同一个对象?
错误的用法:
我看到有些人在使用sys.getrefcount(var)时不知道如何传递var,而是向对象添加引用。一起看看下面的例子。
输出3,而期望的却是2(x andy)。发生这种情况是因为将x传递给getrefcount函数时又添加了一个引用。
from sys import getrefcount x = [1, 2] y = x print(hex(id(x))) # output: 0xb65748 print(hex(id(y))) # output: 0xb65748 print(getrefcount(x)) # output: 3
更好的用法:
可以使用内置的ctypes模块来找到预期的结果。必须将x的id传递给from_address函数。
from ctypes import c_long x = [1, 2] y = x print(hex(id(x))) # output: 0x3395748 print(hex(id(y))) # output: 0x3395748 print(c_long.from_address(id(x)).value) # output: 2
概言之,错误的用法是传递变量,而更好的用法则是传递变量的id,这意味着只传递基数为10的数字,而不是变量。
当对象消失时
当没有变量引用对象时会发生什么?
对象将从内存中删除,因为没有引用该对象的内容。不过也有例外:如果有循环引用,garbage collector 将开始奏效。
为什么使用可变对象
不可变对象由于性能原因,结果可能与预期不同。查看下面的例子,观察输出是如何变化的。
import sys import ctypes """Some Mutable Objects """ a =list() b =set() c =dict() d =bytearray() """ Some ImmutableObjects """ e =tuple() f =int() g =str() print(sys.getrefcount(a),ctypes.c_long.from_address(id(a)).value) # output: 2 1 print(sys.getrefcount(b),ctypes.c_long.from_address(id(b)).value) # output: 2 1 print(sys.getrefcount(c),ctypes.c_long.from_address(id(c)).value) # output: 2 1 print(sys.getrefcount(d),ctypes.c_long.from_address(id(d)).value) # output: 2 1 print(sys.getrefcount(e),ctypes.c_long.from_address(id(e)).value) # output: 1298 1297 print(sys.getrefcount(f),ctypes.c_long.from_address(id(f)).value) # output: 209 208 print(sys.getrefcount(g),ctypes.c_long.from_address(id(g)).value) # output: 59 58
文中所谈及的一切都对CPython有效。希望对你有帮助。
【云栖号在线课堂】每天都有产品技术专家分享!
课程地址:https://yqh.aliyun.com/live立即加入社群,与专家面对面,及时了解课程最新动态!
【云栖号在线课堂 社群】https://c.tb.cn/F3.Z8gvnK
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
如何运行 O’Reilly 书 Python for Finance 的源代码
GitHub 中有一个 https://github.com/yhilpisch/py4fi 项目。 你可以将这个项目 fork 到本地后运行。 项目使用了 jupyter notebook 如果你的系统中没有安装 jupyter notebook 的话,你首先需要安装这个。 Anaconda 的安装方式请参考: https://www.ossez.com/t/anaconda-jupyter-notebook/126 当你安装完成后,如果你使用的是 Windows 的环境。 你需要依次运行: cd py4ficonda env create -f py4fi_conda.ymlactivate py4ficd jupyter36jupyter notebook 最后你会看到 jupyter notebook 启动后,浏览器会打开。 然后你可以直接对代码进行修改运行。 如果你想直接退出的话,你可以单击页面中的 Quite 退出运行的服务器。 服务器运行后,打开网页的默认端口为 8888
- 下一篇
从 React 迁移到 TypeScript:忍受了 15 年的 JavaScript 错误从此走远
云栖号资讯:【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来! Beta 版的 Execute Program 是用 Ruby 和 JavaScript 编写的。之后,我们分几步将整个应用完全移植到了 TypeScript 上。本文介绍的是移植的第一步,也就是前端的部分。 在 Execute Program 的原始 JavaScript 前端中,我经常会犯一些小错误。例如,我会将错误的 prop 名称传递给 React 组件,或者遗漏某个 prop,抑或传递错误的数据类型。(Prop 是作为参数发送到 React 组件的数据。组件将一些 props 传递给自己的某个子组件,以此类推,这是很常见的。) 对于像 JavaScript 和 Ruby 这样的动态语言来说,这不是个小问题。过去 15 年来我一直在学习该如何应对这种错误问题。我在之前谈论的是 2011 年代的情况,其中讨论的缓解措施确实有些用途,但它们无法随着系统的发展顺利地扩展下去,而且我们忘掉它们时也没有安全网可用。 我觉得 15 年时间已经够长了。我想回到静态类型系统的怀抱,毕竟这种系统中...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- MySQL8.0.19开启GTID主从同步CentOS8
- Mario游戏-低调大师作品
- CentOS关闭SELinux安全模块
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- Docker安装Oracle12C,快速搭建Oracle学习环境
- Red5直播服务器,属于Java语言的直播服务器
- CentOS8编译安装MySQL8.0.19
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池