Python中,变量的使用看起来非常简单,例如 a = 10s = "hello"等等。

然而,这种简单的赋值操作背后,CPython其实做了很多复杂的工作。

本文将通过一些简单易懂的代码示例,一起探索Python变量背后的奥秘,让我们对它的实现机制有更深一步的理解。

1. 变量到底是什么?

Python中,变量本质上是一个名字到值的映射。

例如,当你写a = 1时,a是一个名字,而1是一个值。

CPython会将这个名字关联起来,以便你后续可以通过名字访问这个

a = 1
print(a) # 输出:1

这种映射关系是通过一个名为命名空间的结构实现的。

命名空间是一个字典,其中的键是变量名,值是变量对应的对象。

它的定义可参考CPython源码中的Include/internal/pycore_frame.h文件。

typedef struct _PyInterpreterFrame {
// 省略... ...
PyObject *f_globals; /* Borrowed reference. Only valid if not on C stack */
PyObject *f_builtins; /* Borrowed reference. Only valid if not on C stack */
PyObject *f_locals; /* Strong reference, may be NULL. Only valid if not on C stack */
// 省略... ...
}

其中,f_locals 保存局部变量映射,函数执行时,局部变量值存于此;

f_globals 用于全局变量,模块级代码块执行时,f_globals 指向模块全局命名空间字典;

f_builtins 关联内置命名空间。

2. 变量的底层实现:字节码

CPython在执行代码时,会先将代码编译成字节码,然后由虚拟机执行这些字节码。我们可以通过 dis 模块查看代码的字节码。

例如,对于a = 1,字节码如下:

import dis

code = """
a = b
"""
dis.dis(code)

  • LOAD_NAME:从命名空间中加载变量b的值
  • STORE_NAME:将值存储到变量a

这两个指令展示了CPython如何处理变量的读取和赋值。

3. 命名空间与作用域

Python中的变量存储在不同的命名空间中,而这些命名空间又与代码的作用域相关,作用域决定了变量的可见性。

Python有三种主要的作用域:

  1. 局部作用域:函数内部的变量
  2. 全局作用域:模块级别的变量
  3. 内置作用域:包含内置函数和类型的命名空间
x = "global"  # 全局变量

def func():
y = "local" # 局部变量
print(x) # 输出:global
print(y) # 输出:local func()

在这个例子中,x是全局变量,y是局部变量。

如果在函数中尝试访问一个未定义的变量,CPython会按照以下顺序查找:

  1. 局部命名空间(f_locals

  2. 全局命名空间(f_globals

  3. 内置命名空间(f_builtins

如果仍然找不到,就会抛出NameError异常。

4. 不同变量的字节码

CPython为不同作用域的变量提供了不同的字节码指令,以优化性能和实现特定的行为。

4.1. 局部变量

在函数中,局部变量使用LOAD_FASTSTORE_FAST指令。

这些指令直接操作一个数组,而不是字典,因此速度更快。

def func():
a = 1 # STORE_FAST
b = a # LOAD_FAST
return b dis.dis(func)

4.2. 全局变量

全局变量使用LOAD_GLOBALSTORE_GLOBAL指令。

这些指令会直接操作全局命名空间。

x = 1

def func():
global x
x = 2 # STORE_GLOBAL
return x # LOAD_GLOBAL dis.dis(func)

4.3. 闭包变量

当函数嵌套时,内部函数可以访问外部函数的变量。

这些变量称为闭包变量,使用LOAD_DEREFSTORE_DEREF指令。

def outer():
x = 1
def inner():
return x # LOAD_DEREF
return inner dis.dis(outer)

5. 类中的变量

在类定义中,变量的行为与函数不同。

类定义中的变量使用LOAD_NAMESTORE_NAME指令,因为类的命名空间会动态地与全局命名空间交互。

x = "global"

class MyClass:
print(x) # 使用 LOAD_NAME
x = "local"
print(x) # 使用 LOAD_NAME MyClass()

输出:

查看指令的话,可以使用:python.exe -m dis .\cpython-variable.py命令。

如果在类中使用嵌套函数,CPython会使用LOAD_CLASSDEREF指令来处理闭包变量。

class MyClass:
x = "cell"
def method(self):
print(x) # 使用 LOAD_CLASSDEREF MyClass().method()

6. 编译器如何选择指令

CPython的编译器会根据变量的作用域和代码块类型选择合适的字节码指令。

例如:

  • 如果变量是局部变量,编译器会生成LOAD_FASTSTORE_FAST
  • 如果变量是全局变量,编译器会生成LOAD_GLOBALSTORE_GLOBAL
  • 如果变量是闭包变量,编译器会生成LOAD_DEREFSTORE_DEREF

7. 总结

Python变量的实现机制比看起来复杂得多,它涉及到字节码指令、命名空间、作用域以及编译器的决策逻辑。

通过理解这些概念,可以更好地掌握Python的变量行为,尤其是在复杂的作用域场景中。

如果对CPython的实现感兴趣,可以进一步阅读其源码中与变量相关的部分。

『Python底层原理』--CPython的变量实现机制的更多相关文章

  1. 『Python基础-3』变量、定义变量、变量类型、关键字Python基础-3』变量、定义变量、变量类型、关键字

    『Python基础-3』变量.定义变量.变量类型.关键字 目录: 1.Python变量.变量的命名 2.变量的类型(Python数据类型) 3.Python关键字 1. Python 变量.变量的命名 ...

  2. 『Python基础-11』集合 (set)

    # 『Python基础-11』集合 (set) 目录: 集合的基本知识 集合的创建 访问集合里的值 向集合set增加元素 移除集合中的元素 集合set的运算 1. 集合的基本知识 集合(set)是一个 ...

  3. 『Python基础-10』字典

    # 『Python基础-10』字典 目录: 1.字典基本概念 2.字典键(key)的特性 3.字典的创建 4-7.字典的增删改查 8.遍历字典 1. 字典的基本概念 字典一种key - value 的 ...

  4. 『Python基础-9』元祖 (tuple)

    『Python基础-9』元祖 (tuple) 目录: 元祖的基本概念 创建元祖 将列表转化为元组 查询元组 更新元组 删除元组 1. 元祖的基本概念 元祖可以理解为,不可变的列表 元祖使用小括号括起所 ...

  5. 『Python基础-8』列表

    『Python基础-8』列表 1. 列表的基本概念 列表让你能够在一个地方存储成组的信息,其中可以只包含几个 元素,也可以包含数百万个元素. 列表由一系列按特定顺序排列的元素组成.你可以创建包含字母表 ...

  6. 『Python基础-7』for循环 & while循环

    『Python基础-7』for循环 & while循环 目录: 循环语句 for循环 while循环 循环的控制语句: break,continue,pass for...else 和 whi ...

  7. 『Python基础-5』数字,运算,转换

    『Python基础-5』数字,运算,转换 目录 基本的数字类型 二进制,八进制,十六进制 数字类型间的转换 数字运算 1. 数字类型 Python 数字数据类型用于存储数学上的值,比如整数.浮点数.复 ...

  8. 『Python基础-4』字符串

    # 『Python基础-4』字符串 目录 1.什么是字符串 2.修改字符串 2.1 修改字符串大小 2.2 合并(拼接)字符串 2.3 使用乘号'*'来实现字符串的叠加效果. 2.4 在字符串中添加空 ...

  9. 『Python基础-1 』 编程语言Python的基础背景知识

    #『Python基础-1 』 编程语言Python的基础背景知识 目录: 1.编程语言 1.1 什么是编程语言 1.2 编程语言的种类 1.3 常见的编程语言 1.4 编译型语言和解释型语言的对比 2 ...

  10. 『Python基础-12』各种推导式(列表推导式、字典推导式、集合推导式)

    # 『Python基础-12』各种推导式(列表推导式.字典推导式.集合推导式) 推导式comprehensions(又称解析式),是Python的一种独有特性.推导式是可以从一个数据序列构建另一个新的 ...

随机推荐

  1. 基于 .NET 开发的多功能流媒体管理控制平台

    前言 今天大姚给大家分享一个基于 .NET 开发且开源(MIT License)的多功能流媒体管理控制平台:AKStream. 项目介绍 AKStream是一个基于.NET开发且开源(MIT Lice ...

  2. uniapp不介入第三方,Android调用各种权限

    代码: onLaunch: function() { console.log('onLaunch') //监听底部中间菜单的事件 uni.onTabBarMidButtonTap(()=>{ p ...

  3. vue使用高德地图初始化坑

    使用的时候有时候会报除Amap没有定义之外的错如 TypeError: v.w.uh is not a constructor Uncaught TypeError: Cannot read prop ...

  4. Mysql之myisam引擎

    这里是早起整理的myisam优势,因为当时刚毕业那会web1.0时代还没过时,很多的门户网站实际上就只是内容展示的时候,或者发布文章公告的场景.所以对于这样的读多写少的场景,大多数使用的还是myisa ...

  5. QT日志类SimpleQtLogger的简单记录

    在现代软件开发中,日志记录是必不可少的部分.它不仅帮助开发者在调试和维护软件时了解程序的运行状态,还能提供关键的错误信息.对于使用Qt框架开发应用程序的开发者来说,选择一个合适的日志库至关重要.本文将 ...

  6. Dapr-6: Dapr 状态管理构建块

    第 6 章 Dapr 状态管理构建块 The Dapr state management building block | Microsoft Docs 分布式应用程序由一组独立的服务构成.尽管每个服 ...

  7. 中电金信:金Gien乐道 | 6月热门新闻盘点 回顾这一月的焦点事件

  8. AlertManager警报通知 使用webhook 钉钉机器人

    # AlertManager警报通知 使用webhook 钉钉机器人 #启动钉钉webhook服务 #dingtalk webhook docker rm -f dingtalk docker run ...

  9. 【Python】【爬虫】【爬狼】005_爬取数据处理-有的链接缺少了“http”的解决方案(not in)

    我认为这一篇文章是没有意义的,说白了就是判断字符串是否包含指定内容. 爬图片链接的时候,发现了一个问题 有的链接缺了http 像我这种使用HTML表格输出的,本来是可以把图片链接显示为图片的,但是有的 ...

  10. [转]Linux系统下的GCC编译过程、使用命令详解(多文件编译、动态库、静态库)

    Linux系统下的GCC编译过程.使用命令详解(多文件编译.动态库.静态库) [Linux]gcc简介+编译过程 翻译 搜索 复制