【发布时间】:2010-02-03 22:28:41
【问题描述】:
我的问题是:这些模式(如下)起源于哪里?
我(在某处)了解到,对于小整数,Python 有独特的“副本”,如果这是正确的词的话。例如:
>>> x = y = 0
>>> id(0)
4297074752
>>> id(x)
4297074752
>>> id(y)
4297074752
>>> x += 1
>>> id(x)
4297074728
>>> y
0
当我查看int的内存位置时,很早就有一个简单的模式:
>>> N = id(0)
>>> for i in range(5):
... print i, N - id(i)
...
0 0
1 24
2 48
3 72
4 96
>>> bin(24)
'0b11000'
我不清楚为什么选择它作为增量。而且,我根本无法解释256以上的这种模式:
>>> prev = 0
>>> for i in range(270):
... t = (id(i-1), id(i))
... diff = t[0] - t[1]
... if diff != prev:
... print i-1, i, t, diff
... prev = diff
...
-1 0 (4297074776, 4297074752) 24
35 36 (4297073912, 4297075864) -1952
36 37 (4297075864, 4297075840) 24
76 77 (4297074904, 4297076856) -1952
77 78 (4297076856, 4297076832) 24
117 118 (4297075896, 4297077848) -1952
118 119 (4297077848, 4297077824) 24
158 159 (4297076888, 4297078840) -1952
159 160 (4297078840, 4297078816) 24
199 200 (4297077880, 4297079832) -1952
200 201 (4297079832, 4297079808) 24
240 241 (4297078872, 4297080824) -1952
241 242 (4297080824, 4297080800) 24
256 257 (4297080464, 4297155264) -74800
257 258 (4297155072, 4297155288) -216
259 260 (4297155072, 4297155336) -264
260 261 (4297155048, 4297155432) -384
261 262 (4297155024, 4297155456) -432
262 263 (4297380280, 4297155384) 224896
263 264 (4297155000, 4297155240) -240
264 265 (4297155072, 4297155216) -144
266 267 (4297155072, 4297155168) -96
267 268 (4297155024, 4297155144) -120
有什么想法、线索、要看的地方吗?
编辑:24 有什么特别之处?
更新:标准library 具有sys.getsizeof(),当我使用1 作为参数调用它时,它返回24。这是很多字节,但在 64 位机器上,类型、值和引用计数各有 8 个字节。另请参阅 here 和 C API 参考 here。
在 cmets 中使用 Peter Hansen 链接中的“来源”花了一些时间。找不到 int 的定义(除了*int_int 的声明),但我确实找到了:
#define NSMALLPOSINTS 257
#define NSMALLNEGINTS 5
【问题讨论】:
-
“有什么想法、线索和地方可以看吗?”是的。阅读源代码。它会告诉你你想知道的一切。
-
是的,但是源代码就像 C 或者什么的。 :) 我希望有一个 PEP 等价物。
-
PEP 不会指定这样的实现细节。如果你想知道类似这样的东西,它实际上不会影响(理智的)代码,它会在源代码中。
-
不在 PEP 中,但也不在源中:docs.python.org/c-api/int.html#PyInt_FromLong(它的值从 -5 到 256)。