<p>在<code>Include/longintrepr.h</code>中有文档:</p>
<pre><code>/* Parameters of the integer representation. There are two different
sets of parameters: one set for 30-bit digits, stored in an unsigned 32-bit
integer type, and one set for 15-bit digits with each digit stored in an
unsigned short. The value of PYLONG_BITS_IN_DIGIT, defined either at
configure time or in pyport.h, is used to decide which digit size to use.
Type 'digit' should be able to hold 2*PyLong_BASE-1, and type 'twodigits'
should be an unsigned integer type able to hold all integers up to
PyLong_BASE*PyLong_BASE-1. x_sub assumes that 'digit' is an unsigned type,
and that overflow is handled by taking the result modulo 2**N for some N >
PyLong_SHIFT. The majority of the code doesn't care about the precise
value of PyLong_SHIFT, but there are some notable exceptions:
- long_pow() requires that PyLong_SHIFT be divisible by 5
- PyLong_{As,From}ByteArray require that PyLong_SHIFT be at least 8
- long_hash() requires that PyLong_SHIFT is *strictly* less than the number
of bits in an unsigned long, as do the PyLong <-> long (or unsigned long)
conversion functions
- the Python int <-> size_t/Py_ssize_t conversion functions expect that
PyLong_SHIFT is strictly less than the number of bits in a size_t
- the marshal code currently expects that PyLong_SHIFT is a multiple of 15
- NSMALLNEGINTS and NSMALLPOSINTS should be small enough to fit in a single
digit; with the current values this forces PyLong_SHIFT >= 9
The values 15 and 30 should fit all of the above requirements, on any
platform.
*/
</code></pre>
<p><code>int</code>的长度是可变长度部分的长度乘以15/16位-数字要么是<a href="https://github.com/python/cpython/blob/22c526394b2ef51b985873ddbfbcc32c16411919/Include/longintrepr.h#L44" rel="nofollow noreferrer">30 bits in ^{<cd3>}, ^{<cd4>}</a>,要么是<a href="https://github.com/python/cpython/blob/22c526394b2ef51b985873ddbfbcc32c16411919/Include/longintrepr.h#L52" rel="nofollow noreferrer">15 bits in ^{<cd5>}</a>;长对象的结构是</p>
<pre><code>struct _longobject {
PyObject_VAR_HEAD
digit ob_digit[1];
};
</code></pre>
<p>有一个成员<a href="https://docs.python.org/3/c-api/structures.html#c.PyVarObject" rel="nofollow noreferrer">ob_size</a>将以字节表示大小-因此,如果<code>PYLONG_BITS_IN_DIGIT</code>是30,那么<code>ob_digit</code>是一个<code>ob_size / sizeof(uint32_t)</code><code>uint32_t</code>的数组,30位在每个位中是有效的;否则<code>ob_digit</code>是一个<code>ob_size / sizeof(uint16_t)</code><code>uint16_t</code>的数组,15位在每个位中存储。你知道吗</p>
<p>这都是<code>Include/longintrepr.h</code>的一部分,但它们只被<code>#ifndef Py_LIMITED_API</code>揭示!你知道吗</p>