Python 源码 Objects 目录下实现了所有对象的数据类型,包括 int、float、str、list、dict、tuple、set 等等。

PyLongObject 结构定义

int 的相关实现在Objects/longobject.c文件中。对应的结构体是PyLongObject,其定义如下:

typedef struct _longobject PyLongObject; /* Revealed in longintrepr.h */

typedef uint32_t digit; // 无符号 32 位整数

struct _longobject {
    PyObject_VAR_HEAD
    digit ob_digit[1];
};

PyObject_VAR_HEAD就是PyVarObject ob_base;的预处理定义,可以看一下这篇文章Python 内核源码解析:底层基础数据结构 PyObject。ob_digit 是一个数组,数组的长度是 1,这个数组就是存储 int 的值。digit 是一个无符号 32 位整数,也就是说,一个 int 的值,可以用 32 位的无符号整数数组来表示。

为什么用一个数组来存储 int 的值

到这里就产生了一个疑问,这里为什么要用一个数组来表示呢?因为 Python 的 int 是可以无限大的,如果用一个 32 位的无符号整数来表示,那么最大值就是 2^32-1,也就是 4294967295,如果要表示更大的值,就需要用到数组了。字节码没什么差异

>>> a = 123131231231231209830129830128301283012830192
>>> print(a)
123131231231231209830129830128301283012830192
>>> type(a)
<class 'int'>
>>> a+1
123131231231231209830129830128301283012830193

从字节码来看并没有什么区别,那 Python 是怎么把这个大的一个值存储到数组中的呢?

# test.py
a = 123131231231231209830129830128301283012830192
a + 1

python -m dis test.py 
  1           0 LOAD_CONST               0 (123131231231231209830129830128301283012830192)
              2 STORE_NAME               0 (a)

  2           4 LOAD_NAME                0 (a)
              6 LOAD_CONST               1 (1)
              8 BINARY_ADD
             10 POP_TOP
             12 LOAD_CONST               2 (None)
             14 RETURN_VALUE

int 的初始化

_PyLong_New初始化一个 int 对象,参数是数组 obdigit 的长度,也是 obsize 的大小。

_PyObject_InitVar初始化一个对象,参数是对象的地址、对象的类型、对象的大小。相关代码如下:

PyLongObject * _PyLong_New(Py_ssize_t size) {
    ...
    // 初始化
    _PyObject_InitVar((PyVarObject*)result, &PyLong_Type, size);
    //  Py_SET_SIZE(op, size);
    //      ob->ob_size = size;
}

int 的加法运算

两个 int 相加会调用x_add函数,参数是加号左边的值和右边的值,相关代码如下。

z 是两个数值的和,通过_PyLong_New初始化一个 int 对象,这里确保 sizea 是两个值的最大值,即 max(sizea, sizeb),然后 z 的大小为 sizea + 1,这么做是为了防止结果溢出。

计算规则在两个 for 循环中,先计算两个 int 的公共部分,然后再计算剩下的部分。最后返回的时候会调用long_normalize函数,这个函数会把 z 的大小调整为实际需要的大小,也就是去掉前面的 0,最多可以省去一个字节。

/* Add the absolute values of two integers. */

static PyLongObject *
x_add(PyLongObject *a, PyLongObject *b)
{
    Py_ssize_t size_a = Py_ABS(Py_SIZE(a)), size_b = Py_ABS(Py_SIZE(b));
    PyLongObject *z;
    Py_ssize_t i;
    digit carry = 0;

    /* Ensure a is the larger of the two: */
    if (size_a < size_b) {
        { PyLongObject *temp = a; a = b; b = temp; }
        { Py_ssize_t size_temp = size_a;
            size_a = size_b;
            size_b = size_temp; }
    }
    z = _PyLong_New(size_a+1);
    if (z == NULL)
        return NULL;
    for (i = 0; i < size_b; ++i) {
        carry += a->ob_digit[i] + b->ob_digit[i];
        z->ob_digit[i] = carry & PyLong_MASK;
        carry >>= PyLong_SHIFT;
    }
    for (; i < size_a; ++i) {
        carry += a->ob_digit[i];
        z->ob_digit[i] = carry & PyLong_MASK;
        carry >>= PyLong_SHIFT;
    }
    z->ob_digit[i] = carry;
    return long_normalize(z);
}

int 的数据结构

obdigit 每个元素是存储在无符号的 32 位整数上,但是PyLong_SHIFT == 30,意味着存储只用到了 30 位,这是为什么呢?翻阅代码可以看到如下注释。longpow() 函数要求 PyLong_SHIFT 必须是 5 的倍数,30 是满足这个要求的最大值,如果是 16 位的整型,那么PyLong_SHIFT == 15

longpow() requires that PyLongSHIFT be divisible by 5
The values 15 and 30 should fit all of the above requirements, on any platform.

int 的存储结构

int 存储计算算法

123131231231231209830129830128301283012830192 = 
    420275184 * (2**30)**0 + \ 
    700977165  * (2**30)**1 + \
    751460717 * (2**30)**2+\ 
    753924886 * (2**30)**3 + \
    92633642 * (2**30)**4

int 的存储结构

总结

支持大整数是 Python 的一个特性,其数据结构是一个经典的设计,通过数据来拓展数字的存储限制。这个设计值得深入学习、借鉴。

本人非计算机专业自学成为一名程序员,已工作八年,有丰富的摸索、自学经验。热衷于编程语言底层实现原理。通过一些空闲时间阅读源码,记录自己的所学及心得。你的关注和鼓励是对我持续输出分享的动力,感谢,共同进步。

公众号


推荐阅读: