当前位置 : 主页 > 网络编程 > 其它编程 >

python中的引用计数_内存中的Python:Python引用计数指南阿里云开发者社区

来源:互联网 收集:自由互联 发布时间:2023-07-02
云栖号资讯【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯还在等什么快来本文将会为你介绍Pytho 云栖号资讯【点击查看更多行业资讯】 在这里您可以找到不
云栖号资讯【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯还在等什么快来本文将会为你介绍Pytho

云栖号资讯【点击查看更多行业资讯】

在这里您可以找到不同行业的第一手的上云资讯还在等什么快来

本文将会为你介绍Python引用计数演示中使用可变列表对象不过本文不介绍C语言实现细节。

需要注意的是代码段的输出在硬件上可能有所不同。

变量是内存引用

Python中的变量是内存引用。如果输入x [12]时会发生什么[12]是对象。

回想一下一切都是Python中的对象。[12]将在内存中创建。x是[12]对象的内存引用。

来看看下面的例子。可以找到x所引用的内存地址。请务必只使用id(x)它会以10为基数而十六进制函数会将其转换为十六进制。

x [1, 2]

print(hex(id(x))) # output: 0x32ebea8

引用计数

现在已经在内存中创建了一个list对象而且x对该对象进行了引用。那么y[1,2]和yx有什么区别

当输入y[1,2]时它将在内存中创建一个新的list对象并且y将引用它。

x [1, 2]

y [1, 2]

print(hex(id(x))) # output: 0x101bea8

print(hex(id(y))) # output: 0x31a5528

而当输入yx时等同于告诉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

原文发布时间2020-05-31

本文作者读芯术

本文来自“读芯术”了解相关信息可以关注“读芯术”

网友评论