非LL(1)文法到LL(1)文法的等价变换

本文详细介绍了如何将非LL(1)文法转换为LL(1)文法,主要涉及提取左公因子和消除左递归两种方法。提取左公因子用于处理文法中的左公共因子,消除左递归则针对直接和间接左递归。通过这些转换,部分文法可以从非LL(1)变为LL(1),但并非所有文法都能成功转换。同时,转换过程中需要注意去除多余的规则。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们喜欢和需要使用是LL(1)文法,但是我们往往遇到的文法不是LL(1)文法这时候需要将它进行转换转换成LL(1)文法。

这里给出两种方法分别是提取左公因子消除左递归

1 提取左公因子

含有左公共因子的文法
若文法中含有形如: A→ α β| α r 的产生式,称 文法含有左公共因子
显然 , SELECT(A→αβ)∩SELECT(A→αr) ≠ф , 文法不是 LL(1) 文法
提取左公因子基本思想
通过改写产生式来推迟决定,读入足够多的输 入,获得足够的信息之后,再做出正确判断。
提取左公因子
有左因子的文法 A→αβ1 | αβ2     等价于 A →A’( β1 | β2)
提左因子 A →αA’   A→ β1 | β2
一般定义:
A→δβ1 |δβ 2 |…|δβ n 1 2 |…|γ m (每个 γ不 δ 开头)
可改写为 A→δA’|γ1 2 |…|γ m    A’→β1 2 |…|β n
这个左公因子有两种形式:显示和隐式的!!!接下来两个例子会讲述这个问题。
例子1:
文法G[S] S →   aSb | aS | ε
请判断该文法是否为LL(1)文法?
解:该文法不是LL(1)文法。
提左因子
S →   aS(b| ε)    S →   ε
引入S’转换成
S →   aSS’    S’→   b| ε    S →   ε  
注意:这里只是举例子讲述原理定义不去判断是否是LL(1)文法!!!具体判断可以参考 LL(1)文法定义及判别_用编程写诗的博客-优快云博客
例子2:
文法 G[A]
A → ad | Bc
B aA | bB
请判断该文法是否为LL(1)文法?
解:含有隐含左公因子
第一次改写后
G[A] A →   ad | aAc | bBc
B →   aA | bB
我们发现还是有公因子的接着提公因子后
A → aA’| bBc   A’→ d| Ac   B → aA | bB
这个才是最终的结果!!!
消除左公因子小结
经过提取左公因子之后,有些文法从非 LL(1) 文法 变为 LL(1) 文法,有的仍不是 LL(1) 文法。因此,文 法不 含左公共因子只是 LL(1) 文法的必要条件
对于隐含的左公因子,可先等价变换为明确的左公 因子之后,再进行提取。
经过提取左公因子之后,有时会使某些规则变为多 余规则 ,故必须检查提取左公因子之后 的文法,去除多余规则。
个别文法不能在有限步之内提取完左公因子
2 消除左递归
含有左递归的文法
若文法中含有形如
a) A A β(A V N , β∈ (V)*) 的规则
b) A →Bβ B→ A α 的规则
那么
a) 中含有左递归的规则,称为 直接左递归
b) 中虽然没有含左递归的规则, A=> => β A => + A …, 称为 间接左递归
含有左递归的文法不是一个LL(1)文法证明:
以直接左递归为例,若有如下产生式
A→A α  | A→β
其中 , α 和β 为任意语法符号串。
不难证明有下面关系式:
First( β ) First( A α )Select( A→A α)
First( β  ) Select( A→ β  )
由于 A→A α A→ β   Select 集相交,故含有左递归的 文法不是 LL(1) 文法
对于产生式 P→   Pα|β ( 串的特点 βα . . . α)
α不 ε β不 P开头,则可改写规则为:
P βP’
P’αP’| ε
例:S  Sa|b 可改写为:
S→bS’
S’ →aS’| ε
消除直接左递归一般定义
P →   1 |Pα 2 |…|Pα m 1 2 |…|β n
其中每个 α 均不等于 ε ,每个 β不 P 开头,则:
P → β 1 P’|β 2 P’|…|β n P’
P’→ α 1 P’|α 2 P’|…|α m P’|ε
注意:
(1)利用这种方法可以把 直接左递归 都消去,即换成右递归。
(2)但是,这种方法并没有消除整个文法的左递归性。 间接左 递归 仍会存在
消除间接左递归
先将 间接 左递归等价 变换 为等价的 直接 左递归 后再消除左递归
eg:
G[A]
A →   aB | Bb
B →   Ac | d
将规则 1,2 代入 3
先变换成直接左递归
A →   aB | Bb
B →   aBc | Bbc | d
再消除 B 的直接左递归
A →   aB | Bb
B →   aBcB’ | dB’
B’  →   bcB’ |  ε
消除左递归
小结
消除左递归之后,有些文法从非 LL(1) 文法变为 LL(1) 文法,有的仍不是 LL(1) 文法。因此,文法
含左公共因子只是 LL(1) 文法的必要条件。
消除左递归之后,有时会使某些规则变为多余规则, 故必须检查消除左递归之后的文法,去除多余规则 消除左递归的文法在形式上可能不同,但均等价。
本程序的所用的存储结构都是string类型的,最主要的存储文法的数据结构为自定义结构,里面包括一个产生式的部,右部以及select集合,至于终结符的first和follow集合,则是定义了一个string类型的数组进行存储。 本程序的求first,follow,select集合的算法即为书上所介绍的方法,即求first的集合时,只看本产生式,求follow集合时,要进行递归查找一个终结符的所有后跟字符,求select其实就是对first与follow集合的运算,最终根据所有的select集合,便可以判断此文法是否为LL1文法。 对于不是LL1文法的产生式,本程序在判断后进行转换,先进行消除递归,然后提取因子,在这两步的每一步结束之后,都要对产生式进行整合,去掉空存储,去掉无法到达的产生式,将select全部置空。 每进行一次LL1)到LL1)的转换之后,都要对其文法性质进行判断,如果是LL1),则跳出,不是则继续,但是当循环一定次数之后仍不是,程序判定其无法转换,也要跳出。 其中还有对第一个终结字符的右部替换与否进行选择,原因是,有些通过替换就可以很方便的进行转换,这个要通过人为进行输入。 提取因子中也有上一段所说的类似的判断机制,目的是为了防止文法因子无法提取完的情况出现。 最终有三种结果,一种是是LL1文法,一种是不是LL1),但是经过转换变成了LL1),还有一种是经过转换也无法变成LL1)。 输入文本格式样例: A A->ad A->Bc B->aA B->bB
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

用编程写诗

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值