语法树的画法(根据文法求字符串)

2023-12-25 06:20
文章标签 语法 字符串 画法 文法

本文主要是介绍语法树的画法(根据文法求字符串),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1.语法树的画法

2.语法树的短语

3.直接短语(直接到根部)

4.素短语

5.句柄

6.算符优先分析句型


1.语法树的画法

文法G[E]:E->E+E | E*E | (E) | i ,字符串 i+i*i

推导方式有两种最左推导和最右推导(推导的技巧就是逐步靠近字符串的形式)

最左推导(从左往右替换):

E->E+E->i+E->i+E*E->i+i*E->i+i*i

最左推导得到的语法树如下:

最右推导(从右往左替换):

E->E+E->E+E*E->E+E*i->E+i*i->i+i*i

最右推导得出的语法树:

如果两种推导画出来的树相同,则说明他们有无二义性,如果树不同就是有二义性

2.语法树的短语

一个节点的最根部就是这个节点的短语,那么短语有几个呢,就看语法树中有分支的字符有几个,就有几个短语

由上图分析,第一个节点(①)的短语是i+i*i        第二个节点的短语i*i            ③④⑤的短语为i

3.直接短语(直接到根部)

所以上面这棵语法树的直接短语就是i,例如这里的直接短语就是 S,a(注意"("")"不是直接短语)

4.素短语

寻找素短语的前提是在短语中找,这些短语要满足以下条件

(1)至少包含一个终结符

(2)该短语不再包含满足第一个条件的更小的短语

由上面的语法树可以得到的语法为①i+i*i        ②i*i        ③i

①i+i*i,其中包含了满足第一个条件的②

②i*i,包含了满足第一个条件的③

所以素短语只有③

注:有些地方会提到最左素短语,最左素短语就是素短语中位于语法树最左的字符

5.句柄

最左最小的树对应的符号串,或者是直接短语中最左的符号串,在上面的语法树中,句柄就是i

例题1:

这里注意T+T也是素短语,因为他不包含满足第一个条件的短语

例题2:

消除左递归和消除回溯(这一部分比较难,还是要多做题感受下)

消除左递归:

例题1:

E-->E+T|T:

① 将T提前,其他改写为文法E':E->TE'

②剩下的写下来,"|"右边的T由于已经被提出来所以变为\varepsilon(空串):E'--->+TE'|\varepsilon

T-->T*F|F:

①T--->FT'

②*FT'|\varepsilon

F-->(E)|i:不符合左递归的形式,所以照搬即可

例题2:

消除回溯:

例题1:G[A] = A->aAB|a|b

①:将公共左因子"a"提走,另一部分改写为A',没有公共左因子的照抄:A-->aA'|b

②:将改造后的文法A'补齐:A'-->AB|\varepsilon

例题2:

6.算符优先分析句型

若有句型...N_{i}a_{i}...N_{j}a_{j}N_{j+1}....,当a_{i}...N_{j}a_{j}属于句柄时,则 N_{i} 和N_{j+1}也在句柄中,这是由于算符文法的任何句型中均无两个相邻的非终结符,且终结符和非终结符相邻时,含终结符的句柄必含相邻的非终结符,句柄终结符之间的关系如下:

例题:
 

第一步:需要将算符优先分析表得出来

这可以看:算符优先分析的方法

第二步:进行算符优先分析

①默认的#是小于当前输入符号(a)的,所以动作为移入

②接下来继续看栈顶的终结符(a),#<a且a>;所以a就形成了句柄

③对句柄进行归约

由式子可以看到:H->a|(S),可以看到句柄a可以归约为H,但是由于这里是算符优先级分析, 只考虑终结符,所以写任何非终结符是没有关系的,所以这里也可以写T

由于栈中最顶层的终结符为#,又因为#<;所以继续移进

接下来继续下一步的操作:

这里最顶层的终结符a满足,a>+,且"("< "a",所以a是这个整个语句的句柄,继续归约为T

继续下一步操作:

此时最顶层的终结符为+,因为“(”<"+",且“+” > ")",所以+为句柄,又因为,两端有两个非终结符,也在句柄中,所以由T-->T+S|S ,形似T+S,将“T+T”归约为T

由于“(“=”)“,且";" < "(" ,所以这里的"("是句柄,由于公式中含有H->G|(S),所以我们可以将其归约为H,但是这里非终结符是什么没有关系,所以统一归结为T

由于式子中包含S-->S;G|G,这里还是可以归约为S,但是同理我们归约为非终结符T也没有影响

所以可得结论:a;(a+a)G[S]文法中推出来的

但是这里我们注意:a;(a+a)的最右推导,无法推导出a;(a+a),所以a;(a+a)不是G[S]文法的句子

这与算符优先分析中的结论矛盾了,但是最右推导属于规范推导,正确答案是: (不是)

所以算符优先分析可能会错误地接受非法的句子,其推导的过程不规范。

这篇关于语法树的画法(根据文法求字符串)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/534508

相关文章

详解MySQL中JSON数据类型用法及与传统JSON字符串对比

《详解MySQL中JSON数据类型用法及与传统JSON字符串对比》MySQL从5.7版本开始引入了JSON数据类型,专门用于存储JSON格式的数据,本文将为大家简单介绍一下MySQL中JSON数据类型... 目录前言基本用法jsON数据类型 vs 传统JSON字符串1. 存储方式2. 查询方式对比3. 索引

MySQL字符串常用函数详解

《MySQL字符串常用函数详解》本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql字符串常用函数一、获取二、大小写转换三、拼接四、截取五、比较、反转、替换六、去空白、填充MySQL字符串常用函数一、

MySQL 多列 IN 查询之语法、性能与实战技巧(最新整理)

《MySQL多列IN查询之语法、性能与实战技巧(最新整理)》本文详解MySQL多列IN查询,对比传统OR写法,强调其简洁高效,适合批量匹配复合键,通过联合索引、分批次优化提升性能,兼容多种数据库... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析

Python中反转字符串的常见方法小结

《Python中反转字符串的常见方法小结》在Python中,字符串对象没有内置的反转方法,然而,在实际开发中,我们经常会遇到需要反转字符串的场景,比如处理回文字符串、文本加密等,因此,掌握如何在Pyt... 目录python中反转字符串的方法技术背景实现步骤1. 使用切片2. 使用 reversed() 函

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

MySQL 获取字符串长度及注意事项

《MySQL获取字符串长度及注意事项》本文通过实例代码给大家介绍MySQL获取字符串长度及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 获取字符串长度详解 核心长度函数对比⚠️ 六大关键注意事项1. 字符编码决定字节长度2

Springboot3+将ID转为JSON字符串的详细配置方案

《Springboot3+将ID转为JSON字符串的详细配置方案》:本文主要介绍纯后端实现Long/BigIntegerID转为JSON字符串的详细配置方案,s基于SpringBoot3+和Spr... 目录1. 添加依赖2. 全局 Jackson 配置3. 精准控制(可选)4. OpenAPI (Spri

使用Python实现base64字符串与图片互转的详细步骤

《使用Python实现base64字符串与图片互转的详细步骤》要将一个Base64编码的字符串转换为图片文件并保存下来,可以使用Python的base64模块来实现,这一过程包括解码Base64字符串... 目录1. 图片编码为 Base64 字符串2. Base64 字符串解码为图片文件3. 示例使用注意

golang float和科学计数法转字符串的实现方式

《golangfloat和科学计数法转字符串的实现方式》:本文主要介绍golangfloat和科学计数法转字符串的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望... 目录golang float和科学计数法转字符串需要对float转字符串做处理总结golang float

Python如何判断字符串中是否包含特殊字符并替换

《Python如何判断字符串中是否包含特殊字符并替换》这篇文章主要为大家详细介绍了如何使用Python实现判断字符串中是否包含特殊字符并使用空字符串替换掉,文中的示例代码讲解详细,感兴趣的小伙伴可以了... 目录python判断字符串中是否包含特殊字符方法一:使用正则表达式方法二:手动检查特定字符Pytho