语义分析_抽象语法树_反汇编 回忆
  • 上次回顾了一下历史
  • python 是如何从无到有的
  • 看到 Guido 长期的坚持和努力
 

 
添加图片注释,不超过 140 字(可选)
 
  • python究竟是如何理解
  • print("hello")的?
  • 这些ascii字母如何被组织起来执行?
 
纯文本
  • 首先编写Guido的简历
print("1982------Guido in cwi") print("1995------Guido in cnri") print("2000------Guido in beopen") print("2005------Guido in google") print("2012------Guido in dropbox") print("2020------Guido in microsoft")
  • 并保存为Guido.py
生成token流
  • 回到shell之后
  • 从字符流生成token流
 

 
添加图片注释,不超过 140 字(可选)
 
  • 这个过程叫做分词
分词
  • 首先把一个个字符组成词
  • 分析一下哪些字可以组成词
  • 术语叫词法分析(lexical analysis)
 

 
添加图片注释,不超过 140 字(可选)
 
  • 词分析出来之后呢?
组词
  • 词分析出来就是怎么组词的问题
  • 哪些词和哪些词先组合
  • 哪些词和哪些词后组合
 
  • 生成一棵抽象语法树
  • AST(Abstract Syntax Tree)
 

 
添加图片注释,不超过 140 字(可选)
 
  • 我能看看这棵ast树么?
引入ast模块

 
添加图片注释,不超过 140 字(可选)
 
  • 具体怎么做呢?
流程
  • 先把这个ast模块导入(import)进来
  • 第一句就是import ast
  • 回车之后没有任何报错
  • 那就是执行成功了
  • 后面也一样
  • 没有报错就是执行成功了
 

 
添加图片注释,不超过 140 字(可选)
 
  • 然后读取guido.py并送到s
  • 然后对于s进行语法分析(parse)

 
添加图片注释,不超过 140 字(可选)
 
  • 再把分析(parse)的结果进行转储(dump)
  • 看起来有点乱
  • 可以清晰一些么?
 
升级Python
  • 目前lanqiao.cn上面的python是3.8
  • 这个清晰缩进的格式需要在3.9以上完成
  • 需要升级
sudo apt update sudo apt install python3.9
  • 升级之后就可以使用Python3.9了

 
添加图片注释,不超过 140 字(可选)
缩进换行
  • 只能在本地演示一下

 
添加图片注释,不超过 140 字(可选)
 
  • 这个就是把词组成语法树的样子
  • 如何理解这棵树呢?
  • 我们看一个例子
表达式运算
  • 如果给的表达式为 1 2 3

 
添加图片注释,不超过 140 字(可选)
 
  • 结合序为下图

 
添加图片注释,不超过 140 字(可选)
 
  • 前两个先结合
  • 得到的结果作为下一个运算的左操作数
  • 然后和第3个结合
结合序
  • 如果把 第一个* 改成 + 号
  • 其他什么也没加

 
添加图片注释,不超过 140 字(可选)
 
  • 表达式是1 + 2 * 3

 
添加图片注释,不超过 140 字(可选)
 
  • 后两个会先结合
  • 得到的结果 作为下一个运算的 右操作数
  • 然后再和1 进行 加法运算
 
  • 有了 语法树
  • 下一步 要做什么呢?
 
  • 这棵语法树 我们能看懂
  • 但是cpu 需要的是
  • 能执行的 一条条字节码指令
 
 
翻译成 字节码
  • 要把源程序 翻译成字节码 才能执行
  • 字节码 对应着cpu的指令
 
  • 怎么把ast 转化为字节码(指令) 呢?
  • 需要 编译(compile)
 
  • 从一种语言 到 另一种语言
  • 从py文件
  • 到字节码(指令)
  • 就是编译
  • compile
 
 

 
添加图片注释,不超过 140 字(可选)
compile

 
添加图片注释,不超过 140 字(可选)
 
  • 我可以看看这个编译过程么?
指令
  • instruction
  • python3 -m dis Guido.py
  • -m 代表使用模块
  • dis 代表反编译(disassemble)
 

 
添加图片注释,不超过 140 字(可选)
 
  • 我们可以看见
  • 前面是行号
  • 每行对应4条指令
  • LOAD_NAME 装载(函数)名字
  • LOAD_CONST 装载常量
  • CALL_FUNCTION 调用函数
  • POP_TOP 弹栈
 
 
编译结果
  • 先看看这个pyc文件
  • 注意他在__pycache__文件夹下
 

 
添加图片注释,不超过 140 字(可选)
 
  • :%!xxd
  • 把文件转化为字节形态
 

 
添加图片注释,不超过 140 字(可选)
 
  • 这纯纯的机器语言字节形态
  • 实在是看不懂啊
  • 这真的是指令么?
 
  • 究竟什么是指令呢?
指令
  • py文件每行print 对应4条指令
  • LOAD_NAME 装载(函数)名字
  • LOAD_CONST 装载常量
  • CALL_FUNCTION 调用函数
  • POP_TOP 弹栈
 

 
添加图片注释,不超过 140 字(可选)
 
  • 这样 我们 能否找到
  • 4条指令 分别对应的 字节状态值
 
找到对应关系
指令助记符
指令含义
十进制状态
十六进制状态
LOAD_NAME
装载函数名称
101
0x65
LOAD_CONST
装载参数
100
0x64
CALL_FUNCTION
调用函数
142
0x8e
POP_TOP
弹栈返回
1
0x01
  • 可以找对应关系

 
添加图片注释,不超过 140 字(可选)
 
  • 我们从头捋一下
python3 执行过程
  • 不管是python3这个游乐场
  • 还是Guido.py这个python程序
  • 都在我们的硬盘上
 

 
添加图片注释,不超过 140 字(可选)
 
  • 先得把文件从硬盘读到内存
python3 执行的过程大致是这样
  • 先把python3.8这个主解释器
  • 加载到内存中
 
  • 然后 在x86-64的cpu上 执行
  • 模拟出 一台python虚拟机
 

 
添加图片注释,不超过 140 字(可选)
 
  • 准备开始 对py文件 解释执行
先编译
  • 然后把参数 Guido.py 这个需要执行的程序 加载到内存
  • 词法分析 得到 词流(token stream)
  • 语法分析 得到 抽象语法树(Abstract Syntax Tree)
  • 编译 得到 字节码 (byte_code)
 

 
添加图片注释,不超过 140 字(可选)
 
  • 也就是编译后 的pyc文件
解释执行
  • 不过 这个pyc指令文件
  • 是基于python虚拟机的 虚拟cpu的 指令集的
 

 
添加图片注释,不超过 140 字(可选)
 
  • 需要放到 模拟好的 python虚拟机中
  • 一条条指令 进行执行
换句话说
  • 简化版的 hello.py 的执行过程是:
  • 给了 python3 一个参数 Guido.py
  • 使用 python3 这个解释器来解释执行 Guido.py
  • Guido.py中的语句一句句地依次解释执行
 
  • 全解释完成 后
  • 退出python这个程序
  • 把控制权交回到shell
 

 
添加图片注释,不超过 140 字(可选)
 
  • 这些 都是基于 解释器python3的
  • 所谓的 解释器python3
  • 先编译成 python虚拟机的 虚拟指令字节码
  • 然后用 python虚拟机 直接执行虚拟指令
 
 
  • 而解释器(python3) 是
  • 在不同系统 不同架构的cpu语言上 运行的
 

 
添加图片注释,不超过 140 字(可选)
 
  • 那不同的系统、cpu架构
  • python3 为什么 都能正确地解释?
 
总结
  • 这次把py源文件
  • 词法分析 得到 词流(token stream)
  • 语法分析 得到 抽象语法树(Abstract Syntax Tree)
  • 这里确立了优先级
 
  • 编译 得到 字节码 (bytecode)
 
  • 字节码我们看不懂
  • 所以反编译 得到 指令文件(opcode)
  • 指令文件是基于python虚拟机的虚拟cpu的指令集
 
 

 
添加图片注释,不超过 140 字(可选)
 
  • 先从 python3最基础的
  • 变量声明和赋值来看看
  • python虚拟机是如何做的?
 
 


[oeasy]python0135_python_语义分析_ast_抽象语法树_abstract_syntax_tree的更多相关文章

  1. 从零写一个编译器(九):语义分析之构造抽象语法树(AST)

    项目的完整代码在 C2j-Compiler 前言 在上一篇完成了符号表的构建,下一步就是输出抽象语法树(Abstract Syntax Tree,AST) 抽象语法树(abstract syntax ...

  2. 编译器开发系列--Ocelot语言1.抽象语法树

    从今天开始研究开发自己的编程语言Ocelot,从<自制编译器>出发,然后再自己不断完善功能并优化. 编译器前端简单,就不深入研究了,直接用现成的一款工具叫JavaCC,它可以生成抽象语法树 ...

  3. AST抽象语法树

    抽象语法树简介 (一)简介 抽象语法树(abstract syntax code,AST)是源代码的抽象语法结构的树状表示,树上的每个节点都表示源代码中的一种结构,这所以说是抽象的,是因为抽象语法树并 ...

  4. 理解Babel是如何编译JS代码的及理解抽象语法树(AST)

    Babel是如何编译JS代码的及理解抽象语法树(AST) 1. Babel的作用是?   很多浏览器目前还不支持ES6的代码,但是我们可以通过Babel将ES6的代码转译成ES5代码,让所有的浏览器都 ...

  5. JavaScript的工作原理:解析、抽象语法树(AST)+ 提升编译速度5个技巧

    这是专门探索 JavaScript 及其所构建的组件的系列文章的第 14 篇. 如果你错过了前面的章节,可以在这里找到它们: JavaScript 是如何工作的:引擎,运行时和调用堆栈的概述! Jav ...

  6. javascript编写一个简单的编译器(理解抽象语法树AST)

    javascript编写一个简单的编译器(理解抽象语法树AST) 编译器 是一种接收一段代码,然后把它转成一些其他一种机制.我们现在来做一个在一张纸上画出一条线,那么我们画出一条线需要定义的条件如下: ...

  7. 【Static Program Analysis - Chapter 2】 代码的表征之抽象语法树

    抽象语法树:AbstractSyntaxTrees 定义(wiki): 在计算机科学中,抽象语法树(abstract syntax tree或者缩写为AST),或者语法树(syntax tree),是 ...

  8. 抽象语法树(AST)

    AST描述 在计算机科学中,抽象语法树(AST)或语法树是用编程语言编写的源代码的抽象语法结构的树表示.树的每个节点表示在源代码中出现的构造.语法是“抽象的”,因为它不代表真实语法中出现的每个细节,而 ...

  9. Babel(抽象语法树,又称AST)

    文章:https://juejin.im/post/5a9315e46fb9a0633a711f25 https://github.com/jamiebuilds/babel-handbook/blo ...

  10. OpenJDK源码研究笔记(十一):浅析Javac编译过程中的抽象语法树(IfElse,While,Switch等语句的抽象和封装)

    浅析OpenJDK源码编译器Javac的语法树包com.sun.source.tree. 抽象语法树,是编译原理中的经典问题,有点难,本文只是随便写写. 0.赋值语句 public interface ...

随机推荐

  1. PaddleOCR之高性能Go语言实现OCR识别

    最近为了让python语言能够直接调用PaddleOCR的C++的动态链接库,针对本人已经开源的PaddleOCR项目https://gitee.com/raoyutian/paddle-ocrsha ...

  2. 机器学习策略篇:详解超过人的表现(Surpassing human- level performance)

    超过人的表现 讨论过机器学习进展,会在接近或者超越人类水平的时候变得越来越慢.举例谈谈为什么会这样. 假设有一个问题,一组人类专家充分讨论辩论之后,达到0.5%的错误率,单个人类专家错误率是1%,然后 ...

  3. Swift 排查引用循环

    ------------恢复内容开始------------ 一.最近使用RxSwift在多次信号的嵌套中,发现一个对象始终始终无法释放 开始想通过Memory Graph验证是否没有释放,一直报错, ...

  4. no implicit conversion of nil into String

    一.Cocoapod 执行pod install命令时报错 [!] An error occurred while processing the post-install hook of the Po ...

  5. ETL工具-nifi干货系列 第六讲 处理器JoltTransformJSON

    1.处理器作用 使用Jolt转换JSON数据为其他结构的JSON,成功的路由到'success',失败的'failure'.处理JSON的实用程序不是基于流的,因此大型JSON文档转换可能会消耗大量内 ...

  6. TiDB 多集群告警监控-中章-融合多集群 Grafana

    author:longzhuquan 背景 随着公司XC改造步伐的前进,越来越多的业务选择 TiDB,由于各个业务之间需要物理隔离,避免不了的 TiDB 集群数量越来越多.虽然每套 TiDB 集群均有 ...

  7. javascript class 方法的this指向问题

    踩坑记录 JavaScript 的 class 里面有两种定义方法的方式 普通函数(fun1) 箭头函数(fun2) class Obj { func1() { // write some code. ...

  8. FFmpeg开发笔记(三十)解析H.264码流中的SPS帧和PPS帧

    ​<FFmpeg开发实战:从零基础到短视频上线>一书的"2.1.1  音视频编码的发展历程"介绍了H.26x系列的视频编码标准,其中H.264至今仍在广泛使用,无论视频 ...

  9. 如何在Spring Boot框架下实现高效的Excel服务端导入导出?

    前言 Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置 ...

  10. Xilinx-HDF的文件内容

    Xilinx-HDF文件 原文:分享:HDF文件的更多用途 Xilnx Vivado能导出HDF文件,给Xilnx SDK创建软件工程.HDF文件的还可以有更多用途. HDF文件是一个zip文件,可以 ...