编译原理之计算FIRST集合和FOLLOW集合

FIRST集合的求解规则

计算各个文法符号X的FIRST(X)时,不断应用下列规则,直到再没有新的终结符号或者ε可以被加入到任何FIRST集合中为止。

  1. 如果X是一个终结符号,那么FIRST(X) = X。
  2. 如果X是一个非终结符号,且X -> Y1Y2 …Yk是一个产生式,其中k ≥ 1,那么如果对于某个i , a 在FIRST(Yi)中且ε在所有的FIRST(Y1)、FIRST(Y2)、….、FIRST(Yi-1)中,就把a加入到FIRST(X)中。也就是说,Y1…Yi-1 =>* ε。如果多于所有的j = 1,2,3,..,k , ε在FIRST(Yj)中,那么将 ε 加入到FIRST(X)中。比如,FIRST(Y1)中的所有符号一定在FIRST(X)中。如果Y1 不能推导出 ε ,那么,我们就不会再向FIRST(X)中加入任何符号,但是如果Y1 =>* ε ,那么我们就加上FIRST(Y2),以此类推。
  3. 如果X -> ε 是一个产生式,那么将ε 加入到FIRST(X)中。
FOLLOW集合的求解规则

计算所有非终结符号A的FOLLOW(A)集合时,不断应用下列规则,直到再没有新的终结符号可以被加入到任意FOLLOW集合中为止。

  1. 将 $ 放到FOLLOW(S)中,其中S是开始符号,而 $ 是输入右端的结束标记。
  2. 如果存在一个产生式 A -> αBβ , 那么FIRST(β)中除 ε之外的所有符号都在FOLLOW(B)中。
  3. 如果存在一个产生式A -> αB,或存在产生式 A -> αBβ 且 FIRST(β)包含 ε ,那么FOLLOW(A)中的所有符号都在FOLLOW(B)中。

上述的文字规则不免有些抽象和枯燥,不如直接举个栗子来看看吧~

例题

文法G[E]:
1. E -> TE’
2. E’-> +TE’
3. E’-> ε
4. T -> FT’
5. T’-> *FT’
6. T’-> ε
7. F -> (E)
8. F -> a

(1) 求FIRST集合(这里只求非终结符号的FIRST集合)
分析:

FIRST(E): 由E -> TE’可得FIRST(E) = FIRST(T)
FIRST(T): 由T -> FT’可得FIRST(T) = FIRST(F)
FIRST(F): 由F -> (E)和F -> a可得FIRST(F) = { ( , a };
FIRST(T’): 由T’-> *FT’和T’-> ε 可得FIRST(T’) = { * , ε };
FIRST(E’): 由E’-> +TE’和E’-> ε可得FIRST(E’) = { + , ε };

结果:

FIRST(E) = { ( , a };
FIRST(T) = { ( , a };
FIRST(F) = { ( , a };
FIRST(T’) = { * , ε };
FIRST(E’) = { + , ε };

(2) 求FOLLOW集合

分析:

FOLLOW(E): 由F -> (E)可得,FOLLOW(E) = { ) ,$}
FOLLOW(E’): 由E -> TE’和E’-> +TE’可得,FOLLOW(E’) = FOLLOW(E)
FOLLOW(T): 由E -> TE’、E’-> +TE’和E’-> ε 可得,FOLLOW(T) = FIRST(E’) / ε +FOLLOW(E) + FOLLOE(E’)
FOLLOW(T’): 由T -> FT’和T’-> *FT’可得,FOLLOW(T’) = FOLLOW(T)
FOLLOW(F): 由T -> FT’、T’-> *FT’和T’-> ε可得,FOLLOW(F) = FIRST(T’) / ε +FOLLOW(T) + FOLLOW(T’)

结果:

FOLLOW(E) = { ) ,$}
FOLLOW(E’) = FOLLOW(E) = { ) ,$}
FOLLOW(T) = FIRST(E’) / ε +FOLLOW(E) + FOLLOE(E’) = {+ , ) , $}
FOLLOW(T’) = FOLLOW(T) = {+ , ) , $}
FOLLOW(F) = FIRST(T’) / ε +FOLLOW(T) + FOLLOW(T’) = {* , + , ) , $}

小练习

文法G[E]:
1. E -> TE’
2. E’-> +E| ε
3. T -> FT’
4. T’-> T| ε
5. F -> PF’
6. F’ -> *F’| ε
7. P -> (E) | a | b | ∩
求非终结符号的FIRST集合和FOLLOW集合。

思考十分钟再往下看哦~ ~ ~ ~ ~

(1) 求FIRST集合(这里只求非终结符号的FIRST集合)
分析:

FIRST(E): 由E -> TE’可得FIRST(E) = FIRST(T) = { ( , a , b , ∩ }
FIRST(T): 由T -> FT’可得FIRST(T) = FIRST(F) = { ( , a , b , ∩ }
FIRST(F): 由F -> PF’ 可得FIRST(F) = FIRST(P) = { ( , a , b , ∩ };
FIRST(P): 由P -> (E) | a | b | ∩ 可得FIRST(P) = { ( , a , b , ∩ };
FIRST(E’): 由E’-> +E’| ε 可得FIRST(E’) = { + , ε };
FIRST(T’): 由T’-> T| ε 可得FIRST(T’) = FIRST(T) +{ ε };
FIRST(F’): 由F’ -> *F’| ε 可得FIRST(F’) = { * , ε };

结果:

FIRST(E) = { ( , a , b , ∩ }
FIRST(T) = { ( , a , b , ∩ }
FIRST(F) = { ( , a , b , ∩ };
FIRST(P) = { ( , a , b , ∩ };
FIRST(E’) = { + , ε };
FIRST(T’) = { ( , a , b , ∩ , ε };
FIRST(F’) = { * , ε };

(2) 求FOLLOW集合

分析:

FOLLOW(E): 由P -> (E) | a | b | ∩ 和E’-> +E| ε 可得,FOLLOW(E) = FOLLOW(E’) + { ) ,$}
FOLLOW(E’): 由E -> TE’可得,FOLLOW(E’) = FOLLOW(E)
FOLLOW(T): 由E -> TE’ 和 T’-> T| ε可得,FOLLOW(T) = FIRST(E’) / ε +FOLLOW(E’)
FOLLOW(T’): 由T -> FT’可得,FOLLOW(T’) = FOLLOW(T)
FOLLOW(F): 由T -> FT’和T’-> T| ε可得,FOLLOW(F) = FIRST(T’) / ε +FOLLOW(T)
FOLLOW(F’): 由F -> PF’和F’ -> *F’| ε 可得,FOLLOW(F’) = FOLLOW(F)
FOLLOW(P): 由F -> PF’ 和 F’ -> *F’| ε 可得,FOLLOW(P) = FIRST(F’) / ε + FOLLOW(F)

结果:

FOLLOW(E) = { ) ,$ }
FOLLOW(E’) = FOLLOW(E) = { ) ,$ }
FOLLOW(T) = FIRST(E’) / ε +FOLLOW(T’) = { + , ) , $ }
FOLLOW(T’) = FOLLOW(T) = { + , ) , $ }
FOLLOW(F) = FIRST(T’) / ε +FOLLOW(T) = { ( , a , b , ∩ , + , ) ,$ }
FOLLOW(F’) = FOLLOW(F) = { ( , a , b , ∩ , + , ) ,$ }
FOLLOW(P) = FIRST(F’) / ε + FOLLOW(F) = {* , ( , a , b , ∩ , + , ), $ }

First集合Follow集合是在编译原理中用于构造语法分析表的重要概念。 1. First集合: - First集合是对文法产生式右部的一种预测性分析,表示该产生式右部能够推导出的终结符号集合。 - 对于一个非终结符号A,它的First集合包括它的所有产生式右部的第一个终结符号,如果产生式右部可以推导出空串,则还需要包括下一个终结符号。 - First集合计算可以通过以下步骤进行迭代计算,直到不再发生改变: - 如果X是一个终结符号,则First(X) = {X}。 - 如果X是一个非终结符号,则遍历它的每个产生式右部: - 如果右部的第一个符号Y是一个终结符号,则将Y加入到First(X)中。 - 如果右部的第一个符号Y是一个非终结符号,则将First(Y)中的所有终结符号加入到First(X)中,如果Y能够推导出空串,则将下一个符号也加入到First(X)中。 2. Follow集合: - Follow集合是对文法产生式左部非终结符号的一种预测性分析,表示该非终结符号出现在右部中的哪些位置。 - 对于一个非终结符号A,它的Follow集合包括所有能够紧随A出现的终结符号。 - Follow集合计算可以通过以下步骤进行迭代计算,直到不再发生改变: - 将文法开始符号的结束符号($)加入到Follow(开始符号)中。 - 遍历每个产生式右部,对于每个产生式右部的非终结符号A: - 将Follow(A)加入到A后面的非终结符号的Follow集合中。 - 如果A后面跟着一个可以推导出空串的非终结符号B或者是A是右部最后一个符号,则将Follow(A)加入到B的Follow集合中。
评论 16
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值