python中的引用怎么理解_干货,Python引用计数,演示中使用可变列表对象

全文共1372字,预计学习时长7分钟

38dbb6fd5266d016a95d173a55386b0135fa3572.jpeg?token=e5e195bcac2dedfc7279ec336000cfda图源:unsplash

本文将会为你介绍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

18d8bc3eb13533fad2aa792469c0421941345bb2.jpeg?token=f946ff12cb0095821486795e9862cdc5

引用计数

现在已经在内存中创建了一个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

f7246b600c3387447c332875931c66ffd62aa0db.jpeg?token=30b0702d726392c0006534efb534a96e

引用计数的数目

接下来的问题是,有多少变量引用同一个对象?

错误的用法:

我看到有些人在使用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的数字,而不是变量。

当对象消失时

d01373f082025aafd871bf1b3afe1462024f1abd.jpeg?token=8ac1985dbc06ed7133624acf37c3c6cb图源:unsplash

当没有变量引用对象时会发生什么?

对象将从内存中删除,因为没有引用该对象的内容。不过也有例外:如果有循环引用,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

2f738bd4b31c8701641f4b9fe46c21290708ff3d.jpeg?token=9ccdb5ed5ac54f012b1e563f4bc3d232图源:Unsplash

文中所谈及的一切都对CPython有效。希望对你有帮助。

f11f3a292df5e0fee07eb2b00e728bae5edf7231.jpeg?token=f5a0f25efd307ef84e3f5d78b9bf6e33

留言点赞关注

我们一起分享AI学习与发展的干货

如转载,请后台留言,遵守转载规范

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值