1. 文法 G(S):spa
(1)S -> AB递归
(2)A ->Da|εit
(3)B -> cC语法
(4)C -> aADC |ε程序
(5)D -> b|εerror
验证文法 G(S)是否是 LL(1)文法?cas
FIRST集:错误
FIRST(A)→FIRST(Da)→{a,b} b是由DA获得的。void
FIRST(A)→FIRST(ε)→{ε}printf
FIRST(B)→FIRST(cC)→{a,c}
FIRST(C)→FIRST(aADC)→{a}
FIRST(C)→FIRST(ε)→{ε}
FIRST(D)→FIRST(b)→{b}
FIRST(D)→FIRST(ε)→{ε}
FOLLOW集:
FOLLOW(S)→{#}
FOLLOW(A)→{a,b,c,#}
FOLLOW(B)→{#}
FOLLOW(C)→{#}
FOLLOW(D)→{a,#}
SELECT集:
SELECT(A->Da)={b,a}
SELECT(A->ε)={a,b,c,#}
SELECT(B->cC)={c}
SELECT(C->aADC)={a}
SELECT(C->ε)={#}
SELECT(D->b)={b}
SELECT(D->ε)={a,#}
由于:
SELECT(A->Da) ∩SELECT(A->ε)={a,b}
因此文法 G(S)不是 LL(1)文法。
2.(上次做业)消除左递归以后的表达式文法是不是LL(1)文法?
E -> E+T | T
T -> T*F | F
F -> (E) | i
消除左递归:
E -> TE'
E' -> +TE' | ε
T -> FT'
T' ->*FT' | ε
F -> (E) | i
FIRST集:
FIRST(E)→FIRST(T)→FIRST(F)→{ ( , i }
FIRST(E')→{+,ε}
FIRST(T)→FIRST(F)→{ ( , i }
FIRST(T')→{*,ε}
FIRST(F)→{ ( , i }
FOLLOW集:
FOLLOW(E)→{ ) , # } 可由 (E)得出
FOLLOW(E')→{ ) , # } 可由 (TE')得出
FOLLOW(T)→{ + , ) , # } 可由 (T+T)得出
FOLLOW(T')→{ + , ) , # } 可由 (T+FT')得出
FOLLOW(F)→{ * , + , ) , # } 可由 F+F 、F*F 、(i+i)得出
SELECT集:
SELECT(E→TE')={ ( , i }
SELECT(E'→+TE')={+}
SELECT(E'→ε)= FIRST(ε)-{ ε } U FOLLOW(E') = FOLLOW(E')= { ) , # } FIRST(ε)不存在,因此等于FOLLOW(E')
SELECT(T→FT')={ ( , i }
SELECT(T'→*FT')={*}
SELECT(T'→ε)= FIRST(ε)-{ ε } U FOLLOW(T') = FOLLOW(T')={+,),#} FIRST(ε)不存在,因此等于FOLLOW(T')
SELECT(F→(E))={(}
SELECT(F→i)={i}
由于:
SELECT(E'→+TE')∩SELECT(E'→ε)= ∅
SELECT(T'→*FT')∩SELECT(T'→ε)=∅
SELECT(F→(E)) SELECT(F→i)=∅
因此:该文法 G(S)是 LL(1)文法。
3.接2,若是是LL(1)文法,写出它的递归降低语法分析程序代码。
E()
{T();
E'();
}
E'()
T()
T'()
F()
代码:
//在此程序中 E' 改成 E1 , T‘ 改成 T1
void ParseE(){
switch(lookahead){
case (,i: //若是开始时的符号是 (,i 就对 E→TE'进行解析
ParseT(); //对 T 解析
ParseE1(); //对 E' 解析
break;
default: //其余形式开头的都是错误
printf("syntax error \n");
exit(0);
}
}
void ParseE1(){
switch(lookahead){ //对语法 E' -> +TE' | ε 进行解析
case +: //当 E' 开始时的符号是 + 就对 E→+TE'进行解析
MatchToken(+); // 匹配当前终结符和正在扫描的单词符号
ParseT(); // 对 T 解析
ParseE1(); // 对 E'解析
break;
case #,): // 当 E' 开始时的符号是 #,) 就对 E→+TE'进行解析
break; //返回空窜
default: // 其余形式开头的都是错误
printf("syntax error \n");
exit(0);
}
}
void ParseT(){
switch(lookahead){ //对语法 T -> FT' 进行解析
case (,i: //当 T 开始时的符号是 (,i 就对 T -> FT'进行解析
ParseF(); // 对 F 解析
ParseT1(); // 对 T’ 解析
break;
default: // 其余形式开头的都是错误
printf("syntax error \n");
exit(0);
}
}
void ParseE1(){
switch(lookahead){ //对语法 T' ->*FT' | ε 进行解析
case *: //当 T‘ 开始时的符号是 * 就对 T' ->*FT'进行解析
MatchToken(*); // 匹配当前终结符和正在扫描的单词符号
ParseF(); // 对 F解析
ParseT1(); // 对 T' 解析
break;
case #,),+: //当 T' 开始时的符号是 #,),+ 就对 T' ->ε进行解析
break;
default: // 其余形式开头的都是错误
printf("syntax error \n");
exit(0);
}
}
void ParseF(){
switch(lookahead){ //对语法 F -> (E) | i 进行解析
case (: //当 F 开始时的符号是 ( 就对 F -> (E)进行解析
MatchToken((); // 匹配当前终结符和正在扫描的单词符号
ParseE(); // 对 E解析
MatchToken()); // 匹配当前终结符和正在扫描的单词符号
break;
case i: //当 F 开始时的符号是 i 就对 F ->i 进行解析
MatchToken(i); // 匹配当前终结符i
break;
default: // 其余形式开头的都是错误
printf("syntax error \n");
exit(0);
}
}
4.加上实验一的词法分析程序,造成可运行的语法分析程序,分析任意输入的符号串是否是合法的表达式。