栈帧是 Python 虚拟机执行的载体之一,它是一个数据结构,用于存储代码运行时的数据。每个栈帧都有自己的局部变量、操作数栈、异常处理信息等。栈帧是一个栈结构,每个栈帧都有一个指向上一个栈帧的指针,这样就形成了一个栈帧链。

栈帧的创建和销毁是由 Python 虚拟机自动完成的,当一个函数被调用时,就会创建一个栈帧,当函数返回时,就会销毁这个栈帧。

栈帧的数据结构如下:

typedef struct _frame PyFrameObject;

struct _frame {
    PyObject_VAR_HEAD
    struct _frame *f_back;      /* previous frame, or NULL 前一个栈帧 */
    PyCodeObject *f_code;       /* code segment */      // 代码段
    PyObject *f_builtins;       /* builtin symbol table (PyDictObject) 内置符号表,字典 */
    PyObject *f_globals;        /* global symbol table (PyDictObject) 全局符号表 字典 */
    PyObject *f_locals;         /* local symbol table (any mapping)  当前作用域 映射 */
    PyObject **f_valuestack;    /* points after the last local */   // 指向最后一个局部变量的后面
    PyObject *f_trace;          /* Trace function */    // 跟踪函数
    int f_stackdepth;           /* Depth of value stack */  // 值栈深度
    char f_trace_lines;         /* Emit per-line trace events? */ // 每行跟踪事件
    char f_trace_opcodes;       /* Emit per-opcode trace events? */ // 每个操作码跟踪事件

    /* Borrowed reference to a generator, or NULL */
    PyObject *f_gen;        // 生成器

    int f_lasti;                /* Last instruction if called */    // 最后一条指令
    int f_lineno;               /* Current line number. Only valid if non-zero */   // 当前行号
    int f_iblock;               /* index in f_blockstack */ // f_blockstack 的索引
    PyFrameState f_state;       /* What state the frame is in 状态 */   // 栈帧的状态
    PyTryBlock f_blockstack[CO_MAXBLOCKS]; /* for try and loop blocks  */   // try 和 loop 块,长度为 CO_MAXBLOCKS=20
    PyObject *f_localsplus[1];  /* locals+stack, dynamically sized */   // 局部变量和栈,动态地长度
};

每个栈帧都有自己的数据栈,数据栈的长度是动态的,它的长度是局部变量的数量加上栈帧的深度。局部变量的数量是由代码对象的 conlocals 属性决定的,栈帧的深度是由代码对象的 costacksize 属性决定的。

栈帧数据结构

f_code 是代码段,它是一个 PyCodeObject 对象,它包含了代码对象的所有信息,包括字节码、常量表、变量名等。

f_builtins 是内置符号表,它是一个 PyDictObject 对象,它包含了内置函数的名称和函数对象的映射关系。

f_globals 是全局符号表,它是一个 PyDictObject 对象,它包含了全局变量的名称和变量值的映射关系。

f_locals 是局部符号表,它是一个 PyDictObject 对象,它包含了局部变量的名称和变量值的映射关系。

f_valuestack 是值栈,它是一个 PyObject 对象的数组,它包含了局部变量和操作数栈的值。

a = 1
b = 3
c = a + b
print(c)

操作码调用

Python 的调用栈有多个栈帧组成,一定包含 main 栈帧,main 栈帧是 Python 虚拟机启动时创建的,它的 fback 指针为 NULL,其它栈帧的 fback 指针都指向上一个栈帧。

def bar(y):
    x = y + 3
    return x

def foo():
    a = 1
    b = 2
    return a + bar(b)

foo()

main 栈帧调用 foo 栈帧,foo 栈帧调用 bar 栈帧,bar 栈帧返回到 foo 栈帧,foo 栈帧返回到 main 栈帧。每一个返回都会销毁一个栈帧。程序结束,main 栈帧也会被销毁。

调用栈


推荐阅读: