来自 sys.getrefcount 的意外值

问题描述 投票:0回答:2

在Python 2.7.5下

>>> import sys
>>> sys.getrefcount(10000)
3

三个refcount在哪里?

PS:当 10000 个 PyIntObject 将 Py_DECREF 为 0 ref 并释放时? 别说gc的东西了,引用计数本身不用gc也能工作。

python python-internals
2个回答
8
投票
  1. 当您在 REPL 控制台中执行某些操作时,字符串将在内部编译,并且在编译过程中,Python 将创建一个中间列表,其中包含除标记之外的字符串列表。所以,这是参考号 1。你可以这样检查

    import gc
    print gc.get_referrers(10000)
    # [['sys', 'dis', 'gc', 'gc', 'get_referrers', 10000], (-1, None, 10000)]
    
  2. 由于它只是一个数字,因此在编译过程中,Python 的窥孔优化器会将数字存储为生成的字节码中的常量之一。你可以这样检查

    print compile("sys.getrefcount(10000)", "<string>", "eval").co_consts
    # (10000,)
    

注:

Python 在列表中存储 10000 的中间步骤仅适用于编译后的字符串。这不是为已编译的代码生成的。

print eval("sys.getrefcount(10000)")
# 3
print eval(compile("sys.getrefcount(10000)", "<string>", "eval"))
# 2

在第二个示例中,我们使用

compile
函数编译代码,并仅将代码对象传递给
eval
函数。现在只有两个参考。一个来自窥孔优化器创建的常量,另一个是
sys.getrefcount
中的常量。


0
投票

sys.getrefcount() 的返回值 3 通常表示:

  • 来自 getrefcount() 调用中临时参数的一个引用。
  • 加上程序中其他地方的两个额外参考。
  • 此计数准确反映了当时的总参考文献 但请记住包含临时参考。
© www.soinside.com 2019 - 2024. All rights reserved.