解释运行程序

回忆上次内容

  • py 文件的程序是按照顺序

    • 一行行挨排解释执行的
    • 我们可以 python3 -m pdb hello.py 来对程序调试
    • 调试的目的是去除 bug
    • 别害怕 bug
    • bug 会有提示
    • 我们也就知道如何 debug 调试
  • 顺序执行

    • 程序在文本中从上到下是一行行写的
    • 调试的时候也是从头到尾一行行执行的
    • 但是执行的时候是如何把代码一行行解释执行的呢?
  • 说到底 python3 到底是个啥呢?

    • python3 又是怎么解释 hello.py 的?
    • 这两这节课相当复杂
    • 如果感觉太过复杂
    • 可以直接跳过
    • 不影响后面的理解
  • 我们先要看看python3对Guido.py做了些什么???

tokenize

  • 首先把字符分组成词
  • 词法分析(lexical analysis)中

编辑

  • 把原来的字符流
  • 变成了词的流

    • token(令牌)流

编辑

  • 词法分析之后输出的是一个token流
  • 什么是token流呢?

    • 首先要知道什么是token

token

  • token

    • 令牌

编辑

  • 古人说听我号令

    • 号指的是号角
    • 令指的是令牌

      • 急急如律令
      • 令行禁止
  • 怎么把源文件变成一个token流呢?

python3模块

  • 这个东西是python3的一个模块

正在上传…重新上传取消

  • 具体怎么运行呢?

token流

  • 我们尝试运行

    • python3 -m tokenize guido.py
    • 对guido.py进行词法分析
  • 分析出来的词(token)流长什么样子呢?

编辑

  • 这个词怎么理解呢?

token流

  • 第0行设置了编码格式
  • 第1行[0,5)字符是第1行第1个token

    • print
    • print是一个Name(名字)
  • 第1行[5,6)字符是第1行第2个token

    • (
    • (是一个Operator(操作符)
  • 第1行[6,30)字符是第1行第3个token

    • "1982------Guido in cwi"
    • 这是一个String(字符串)
  • 第1行[30,31)字符是第1行第4个token

    • )
    • )是一个Operator(操作符)
  • 第1行[31,32)字符是第1行第5个token

    • \n
    • \n是一个NewLine(换行符)
    • 换行符意味着第一行结束
  • 第2行...

编辑

  • 词分析出来之后呢?

组词

  • 词分析出来就是怎么组词的问题

    • 哪些词和哪些词先组合
    • 哪些词和哪些词后组合
  • 生成一棵抽象语法树

    • AST(Abstract Syntax Tree)

编辑

  • 具体怎么生成这棵ast树呢?

引入ast模块

编辑

  • 具体怎么做呢?

流程

  • 先把这个ast模块导入(import)进来
  • 然后读取guido.py并送到s
  • 然后对于s进行语法分析(parse)

编辑

  • 不过这乱七八糟堆一起怎么理解呢?

缩进换行

  • 把分析的结果进行dump(转储)

编辑

  • 目前lanqiao.cn上面的python是3.8
  • 这个换行需要在3.9以上完成
  • 只能在本地演示一下

缩进演示

编辑

  • 这个就是把词组成语法树的样子
  • 但是语法树还不能直接执行
  • 什么才能直接执行呢?

翻译成字节码

  • 字节码(指令)才能真正执行
  • 怎么把ast转化为字节码(指令)呢?
  • 需要编译
  • 从一种语言到另一种语言

    • 从py文件
    • 到字节码(指令)

编辑

  • 我可以看看这个编译过程么?

compile

编辑

编辑

  • 这个东西完全是乱码
  • 我看不懂啊?
  • vi打开这个这个pyc文件

二进制形态

  • :set wrap设置换行

编辑

  • 可以看到他的二进制形态么?

二进制

  • :%!xxd

    • 把文件转化为二进制

编辑

  • 实在是看不懂啊
  • 能把这个字节码(指令)变成我们人能看懂的么?

反编译

  • disassembler这个词由两部分组成

    • dis (反着来的)
    • assembler (汇编语言)
  • 整体就是

    • 把py源文件编译成的字节码(指令)
    • 反编译(disassembler)成这些字节码对应的助记符(指令的含义)

编辑

  • 这可以用么?
  • 去试试!

反编译(dis)

  • python3 -m dis guido.py

编辑

  • 我们可以看见

    • 前面是行号
    • 每行对应4条指令

      • LOAD_NAME 装载函数名
      • LOAD_CONST 装载参数
      • CALL_FUNCTION 调用函数
      • POP_TOP 弹栈返回
    • 每条指令对应一个字节码
  • 那具体这个LOAD_NAME是什么意思呢?

指令

  • 指令对应着一个字节码状态

编辑

  • 但是LOAD_NAME这条指令
  • 具体对应什么二进制字节状态呢?

二进制状态

编辑

  • 我们找找程序中的4条指令对应的字节状态

4条指令

指令助记符 指令含义 十进制状态 十六进制状态
LOAD_NAME 装载函数名称 101 0x65
LOAD_CONST 装载参数 100 0x64
CALL_FUNCTION 调用函数 142 0x8e
POP_TOP 弹栈返回 1 0x01
  • 可以找到源代码的对应关系么?

编辑

  • 好像找到了
  • 但是0x83 对应的是 GET_AWAITABLE
  • 显然00 83是从表中的0号位置取得字符串变量
  • 01 83是从表中的1号位置取字符串
  • 以此类推,直到05 83
  • 那这些代码究竟是什么指令集的呢?

    • 龙芯
    • intel
    • 还是arm呢?

虚拟机的虚拟cpu

  • 这些字节码(bytecode)对应的是python虚拟机上面虚拟cpu的指令集

编辑

  • 怎么还有虚拟机
  • 虚拟cpu呢?
  • 我们先把这节课总结一下

总结

  • 我们把python源文件

    • 词法分析 得到 词流(token stream)
    • 语法分析 得到 抽象语法树(Abstract Syntax Tree)
    • 编译 得到 字节码 (bytecode)
    • 反编译 得到 指令文件

编辑

  • 不过这个指令文件是基于虚拟机的虚拟cpu的指令集
  • 怎么这么虚呢?
  • 我们下次再说

本文章来自于《oeasy教您玩转python》(https://www.lanqiao.cn/courses/3584)中第6个实验。

python代码是如何执行的?的更多相关文章

  1. JAVA,Python代码是编译执行还是解释执行?

    转载地址:http://blog.csdn.net/zv3e189os5c0tsknrbcl/article/details/78661641 有人在讨论 Python 代码是编译执行还是解释执行?这 ...

  2. 关于python代码是编译执行还是解释执行

    Python 是编译型语言还是解释型语言?回答这个问题前,应该先弄清楚什么是编译型语言,什么是解释型语言. 所谓编译执行就是源代码经过编译器编译处理,生成目标机器码,就是机器能直接运行的二进制代码,下 ...

  3. 交互模式下测试python代码及变量的四则运算

    在交互模式下,python代码可以立即执行,所以这很方便我们进行代码测试 1.命令窗口,输入python (如果没配置环境变量则需带python安装目录的绝对路径) >>> 这个就是 ...

  4. 深入理解 GIL:如何写出高性能及线程安全的 Python 代码

    深入理解 GIL:如何写出高性能及线程安全的 Python 代码 本文由 伯乐在线 - 郑芸 翻译.未经许可,禁止转载!英文出处:A. Jesse.欢迎加入翻译组. GIL对多线程的影响:http:/ ...

  5. python代码块,小数据池,驻留机制深入剖析

    一,什么是代码块. 根据官网提示我们可以获知: 根据提示我们从官方文档找到了这样的说法: A Python program is constructed from code blocks. A blo ...

  6. 一种部署 Python 代码的新方法

    在Nylas,我们喜欢使用Python进行开发.它的语法简单并富有表现力,拥有大量可用的开源模块和框架,而且这个社区既受欢迎又有多样性.我们的后台是纯用 Python 写的,团队也经常在 PyCon ...

  7. 用Pylint规范化Python代码,附PyCharm配置

    Pylint一个可以检查Python代码错误,执行代码规范的工具.它还可以对代码风格提出建议. 官网:https://pylint.readthedocs.io pip install pylint ...

  8. Python代码的编译

    Python代码的编译 Python代码在解释执行之前,是会被编译成.pyc或者.pyo文件的,它们是中间字节码表示的文件,之后Python虚拟机才会去解释执行它们. 1.pyc文件 ======== ...

  9. Python基础:27执行环境

    一:可调用对象 可调用对象,是任何能通过函数操作符“()”来调用的对象.Python 有4 种可调用对象:函数,方法,类,以及一些类的实例. 1:函数 python 有 3 种不同类型的函数对象. a ...

随机推荐

  1. 使用C#跨PC 远程调用程序并显示UI界面

    在项目中有一个需求是需要在局域网内跨PC远程调用一个程序,并且要求有界面显示,调查了一些资料,能实现远程调用的.Net技术大概有PsExec.WMI.Schedule Task. 这三种方式都做了一个 ...

  2. Swift初探02 数组、集合、元组、字典

    数组.集合.元组.字典 每一门语言基本都会有这些数据结构,swift也不例外,而作为一门现代的语言,swift提供了很多的现成的方法给我们选择. 一.数组 01 数组的定义 // 三种声明方式 var ...

  3. (数据科学学习手札136)Python中基于joblib实现极简并行计算加速

    本文示例代码及文件已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes 1 简介 我们在日常使用Python进行各种数据计算 ...

  4. JavaScript 模块的循环加载(循环依赖问题分析)

    简介 "循环加载"(circular dependency)指的是,a 脚本的执行依赖 b 脚本,而 b 脚本的执行又依赖 a 脚本. 分析 使用 madge 工具进行循环加载分析 ...

  5. grpc-java源码环境编译

    1. Clone 1.1 git clone https://github.com/grpc/grpc-java.git 1.2 idea 打开grpc-java工程 2.compile 2.1 ja ...

  6. CefSharp 白屏问题

    原文 现象 我正在使用 cefsharp + winform 建立一个桌面程序用于显示网页.使用过程中程序会突然白屏,经过观察发现,在网页显示GIF动图时,浏览器子程序会突然占用较高内存(从80M上升 ...

  7. Es6语法+v-on参数相关+vue虚拟dom

    Es6的语法 Es5:if和for 都没有块级作用域,函数function有作用域. Es6:加入let使得if和for有作用域 .建议: 在Es6中优先使用const,只有需要改变某一个标识符的时候 ...

  8. 机器学习-学习笔记(二) --> 模型评估与选择

    目录 一.经验误差与过拟合 二.评估方法 模型评估方法 1. 留出法(hold-out) 2. 交叉验证法(cross validation) 3. 自助法(bootstrapping) 调参(par ...

  9. 本地创建的jupyter notebook 无法连接本地环境(即不能运行代码)

    参考:https://www.cnblogs.com/damin1909/p/12691147.html 本人所用的python是anaconda下的,由于需求不同,创建了好多个python用于不同的 ...

  10. UVA471 Magic Numbers 题解

    1.题目 题意很简单:输入n,枚举所有的a,b,使得 (1)满足a/b=n. (2)满足a,b各个位上的数字不相同. 2.思路 (1)对于被除数,要满足各个位上的数字,显然最大枚举到987654321 ...