美文网首页iOS学习
编译器学习之 (五) : 语义分析

编译器学习之 (五) : 语义分析

作者: sea_biscute | 来源:发表于2017-09-28 20:44 被阅读327次

前言

生成抽象语法树之后,下一步要做的就是对其进行分析,这个过程就称为语义分析,在此步骤我们需要做的有

  1. 变量引用的消解
  2. 类型名称的消解
  3. 类型定义检查
  4. 表达式的有效性检查
  5. 静态类型检查

以上的五个过程的执行顺序有着一定的限制.就是类型名称的消解类型定义检查的前提,而表达式的有效性检查在前三个过程结束前是不能执行的,介于它们执行顺序的关系,可以由下图表示

语义分析处理顺序

变量引用的消解

变量引用的消解是指将所有变量和它们的定义关联起来,例如变量a可能是全局变量a,也可能是静态变量a,还可能是局部变量a,为了消除这种不确定性,我们对它们进行和定义的关联.
具体操作就是给变量节点对象增加变量定义的属性.

类型名称的消解

在有的语言中,类型名称TypeRef类型实体Type是分开处理的,原因是TypeRef可以理解为类型的声明,Type则是类型的实现,我们有可能在实现之前就使用了这种类型.所以才会区分处理.
这里的消解,是将TypeRefType进行关联.通过一个TypeTable的对象去管理它们的对应关系.

类型定义的检查

类型定义的检查是检查在定义类型时使用了不符合逻辑的定义声明,比如

  1. 包含 void 的数组、结构体、联合体
  2. 成员重复的结构、联合体
  3. 循环定义的结构体、联合体

1和2的实现非常简单,这里就不赘述了,3可以这样来理解,结构体或者联合体对其他类型是引用可以看做是一个有向使用,相互之前的关系可以做成一个图

无闭环 有闭环

如果存在闭环,则存在循环定义.

检测有向图中的闭环的算法

要检测有向图是否存在闭环,可以使用如下算法.

  1. 选择任意一个节点并标注为查找中
  2. 沿着边依次访问所有与该节点相邻的节点
  3. 如果访问到的节点没有标注任何状态,则将节点标注为查找中;如果标注了``查找结束,则不做任何处理,返回之前的节点;如果标注为查找中`,则说明存在闭环.
  4. 从当前的节点重复步骤2和3,如果已经没有可访问的相邻节点,则将该节点标注为查找结束,并沿原路返回.
  5. 按照上述流程对所有节点进行处理,如果没有遇到查找中状态的节点,就说明不存在闭环.

表达式有效性检测

表达式需要检测的问题有

为无法赋值的表达式赋值(例:1 = 2 + 2)
使用非法的函数名调用函数(例:"string"("%d\n", i))
操作数非法的数组引用(例:1[0])
操作数非法的成员引用(例:1.memb)
操作数非法的指针间接引用(例:1->memb)
对非指针的对象取值(例:*1)
对非左值的表达式取地址

静态类型检查

语言的操作都对操作数的类型有所限制.例如结构体之间无法用 + 进行 加法运算,指针和数值之间无法用 * 进行乘法运算,将数组传递给参数类型为 int 型的函数会出现莫名其妙的结果.

对允许的操作数类型进行限制,例如 * 操作只适用于类型相同的数值之间,在编译过程中检查是否符合这样的限制的处理就是静态类型检查(static type checking).

静态类型检测的隐式类型转换

二元运算 * 只允许在相同类型的整数之间进行,但是如果在不同类型的数值之间进 行 * 运算,为了能够正常运算,编译器会自动对操作数的类型进行转换.这样的转换就称为隐式类型转换(implicit conversion).
例如,当int类型的值和long类型的值进行乘法运算时, 编译器会将两个操作数统一为 long 类型.

另外,在一些不得不强行转换为特定类型的情况下也会发生隐式类型转换.
例如,当 return 返回值的类型和函数返回值的类型不一致时,必须转换为函数返回值的类型
赋值运算时必须转换为和左值一致的类型等。显式声明的函数返回值或变量类型无法轻易改变,因此 只能利用隐式转换来改变值的类型。
在静态类型检查过程中也会实施隐式类型转换

相关文章

  • 编译器学习之 (五) : 语义分析

    前言 生成抽象语法树之后,下一步要做的就是对其进行分析,这个过程就称为语义分析,在此步骤我们需要做的有 变量引用的...

  • 静态链接

    编译和链接 预处理 编译扫描(词法分析)、语法分析、语义分析(静态语义是编译器所能分析的,动态语义要在运行期才能确...

  • iOS 编译过程

    iOS 编译采用 Clang 作为编译器前端,LLVM 作为编译器后端,编译器前端负责语法分析,语义分析,生成生成...

  • 编译器---语义分析(java)

    语义分析的任务: 负责检查抽象语法树的上下文相关属性: 变量使用前,需要事先定义 变量运算时,类型需要匹配 变量的...

  • 编译过程做了哪些事情

    编译器结构clang是编译器前端,llvm是编译器后端,clang主要做一些语法语义分析, 如果有错误在这里就发现...

  • 编译器的工作过程

    编译器的工作过程划分为:词法分析、语法分析、语义分析、中间代码生成、代码优化和目标代码生成。 词法分析器的任务是把...

  • 从点击运行到显示画面经历的那些事

    只是介绍重点过程 预编译器符号化宏定义的展开import的展开 编译器语法和语义分析将符号化后的内容转化为一棵解析...

  • 编译原理概述

    编译器原理 词法分析器 语法分析器 语义分析器 中间代码生成 符号表 独立机器的代码优化器 代码生成器 依赖于机器...

  • 大数据语义分析_大数据语义理解_语义分析解决方案_ImageQ

    针对大数据语义分析、大数据语义理解延伸出的各行业语义分析解决方案,基本上ImageQ语义分析平台都能满足到位,对于...

  • 统计学习方法——修炼学习笔记20:潜在狄利克雷分配

    潜在狄利克雷分配LDA,作为基于贝叶斯学习的话题模型,是潜在语义分析、概率潜在语义分析的扩展。在文本数据挖掘、图像...

网友评论

    本文标题:编译器学习之 (五) : 语义分析

    本文链接:https://www.haomeiwen.com/subject/rhcpextx.html