最左推导和最右推导,语法树,二义性文法

※最左推导和最右推导

(每一步替换最左边的非终结符/每一步替换最右边的非终结符),最右推导称为规范推导。最右推导对应于最左规约(规范规约)

例:

文法:

S--->AB

A--->a|t

B---->+CD

C--->a

D---->a

最右推导:

S--->AB---->A+CD--->A+Ca---->A+aa----->a+aa

最左推导:

S---->AB----->aB--->a+CD--->a+aD----->a+aa



※语法树:

文法:

最左推导和最右推导,语法树,二义性文法 - 小镜子~ - 菜园子
 

相应的语法树:

最左推导和最右推导,语法树,二义性文法 - 小镜子~ - 菜园子
 

(这是最右推导的语法树)



※文法的二义性:

给定一个文法G,如果LG)中存在一个具有两棵或两棵以上分析树的句子, 我们就称该文法为二义性的,G也叫二义性文法。

对于以下文法:

最左推导和最右推导,语法树,二义性文法 - 小镜子~ - 菜园子
 

对于id+id*id的串,有两个分析树与之对应

最左推导和最右推导,语法树,二义性文法 - 小镜子~ - 菜园子
 

造成二义性的原因是:文法中没有体现出结合率和优先级

### Java 中推导推导概念 在编译原理中,对于给定的文法起始符号,可以通过一系列替换操作来生成特定字符串的过程称为推导。当涉及到Java编程语言时,理解推导推导有助于更好地掌握其解析机制。 #### 推导 (Leftmost Derivation) 推导是指每次总是先处理当前串中边的第一个非终结符,并将其替换成相应的产生式的侧部分[^3]。这种策略确保了每一步都优先扩展侧的内容直到整个表达式被完全展开为止。 #### 推导 (Rightmost Derivation) 相反地,在推导过程中,每次都选择边的一个非终结符来进行替换工作。这种方式使得终形成的派生序列是从逐步建立起来的。 ### 构建语法树的方法 为了构造一棵表示源代码结构化的抽象语法树(AST),通常采用自顶向下或自底向上两种主要方式: - **自顶向下**:从根节点开始尝试匹配输入流中的标记(token),并按照预定义好的规则不断创建新的分支直至叶节点代表具体的词素。 - **自底向上**:则是在扫描到一定数量的有效token组合成更高级别的单元后再回溯形成父级节点,后汇聚于单个根部位置完成整棵树形图谱搭建。 以递归下降分析为例,这是一种典型的自顶向下方法,它为每个非终端编写一个过程,该过程负责识别由相应非终端产生的句子形式。如果遇到错误情况如`int a;`而非预期的`int main()`,将会触发异常报告指出具体问题所在之处。 ```java // 假设有一个简单的E->T|E+T T->F|T*F 的算术表达式文法规则, // 下面展示了一个基于这些规则的部分实现片段用于说明如何进行/推导以及构建AST: public class Parser { private Token currentToken; public Node parseExpression() { // 对应 E -> T | E + T Node left = parseTerm(); while (currentToken.type == PLUS) { advance(); // 跳过 '+' 符号 Node right = parseTerm(); left = new PlusNode(left, right); // 创建加法运算节点作为新父节点连接子树 } return left; } private Node parseTerm() { ... } // 类似地定义其他辅助函数... } ```
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值