提问人:holamynameisyes 提问时间:11/2/2023 更新时间:11/2/2023 访问量:36
减少/减少冲突野牛语法
Reduce/reduce conflict bison grammar
问:
我是野牛解析的新手,我不完全了解它是如何工作的。我有以下简单的野牛语法来解析一个简单的语言:
%{
%}
%token T_ASSIGN T_ADD T_SUB T_MUL T_DIV T_MOD T_POW
%token T_EQ T_NE T_LT T_GT T_LE T_GE T_AND T_OR T_NOT
%token T_ENDLINE T_LPAR T_RPAR
%token T_INTEGER T_FLOAT T_BOOL
%token T_ID
%%
program: statement_list
statement_list:
statement_list T_ENDLINE
statement
| statement
statement: %empty | expr
expr: arthm_expr | bool_expr | assignment
arthm_expr:
arthm_expr sum_op mul
| mul
sum_op: T_ADD | T_SUB
mul:
mul mul_op power
| power
mul_op: T_MUL | T_DIV | T_MOD
power:
armth_last T_POW power
| armth_last
armth_last:
T_INTEGER
| T_FLOAT
| T_ID
| T_LPAR arthm_expr T_RPAR
assignment:
T_ID T_ASSIGN expr
bool_expr:
bool_expr T_OR bool_and
| bool_and
bool_and:
bool_and T_AND bool_cmp_eq
| bool_cmp_eq
bool_cmp_eq:
arthm_expr eq_op arthm_expr
| bool_cmp
eq_op: T_EQ | T_NE
bool_cmp:
arthm_expr cmp_op arthm_expr
| bool_unary
cmp_op: T_LT | T_GT | T_LE | T_GE
bool_unary:
T_NOT bool_unary
| bool_last
bool_last:
T_BOOL
| T_ID
| T_LPAR bool_expr T_RPAR
%%
如您所见,可以采用布尔表达式和算术表达式。T_ID
当我编译语法时,它会产生以下 reduce/reduce 冲突(提取 fom bison 输出):
example.y: warning: 3 reduce/reduce conflicts [-Wconflicts-rr]
example.y: warning: reduce/reduce conflict on tokens $end, T_ENDLINE [-Wcounterexamples]
Example: T_ID •
First reduce derivation
expr
↳ 6: arthm_expr
↳ 10: mul
↳ 14: power
↳ 19: armth_last
↳ 22: T_ID •
Second reduce derivation
expr
↳ 7: bool_expr
↳ 26: bool_and
↳ 28: bool_cmp_eq
↳ 30: bool_cmp
↳ 34: bool_unary
↳ 40: bool_last
↳ 42: T_ID •
example.y: warning: reduce/reduce conflict on token T_RPAR [-Wcounterexamples]
Example: T_LPAR T_ID • T_RPAR
First reduce derivation
expr
↳ 6: arthm_expr
↳ 10: mul
↳ 14: power
↳ 19: armth_last
↳ 23: T_LPAR arthm_expr T_RPAR
↳ 10: mul
↳ 14: power
↳ 19: armth_last
↳ 22: T_ID •
Second reduce derivation
expr
↳ 7: bool_expr
↳ 26: bool_and
↳ 28: bool_cmp_eq
↳ 30: bool_cmp
↳ 34: bool_unary
↳ 40: bool_last
↳ 43: T_LPAR bool_expr T_RPAR
↳ 26: bool_and
↳ 28: bool_cmp_eq
↳ 30: bool_cmp
↳ 34: bool_unary
↳ 40: bool_last
↳ 42: T_ID •
如何
答:
2赞
Chris Dodd
11/2/2023
#1
冲突的发生是因为您的语法模棱两可——当输入是没有运算符字符的简单输入时,就无法判断它应该被解析为 a 还是 .T_ID
bool_expr
arithm_expr
基本上有两种方法可以解决这个问题
不要在语法中打字 -- 去掉 and 之间的区别,只用表达式在它们之间可以有任何运算符。在解析后,您可能需要一个语义类型检查传递(访问解析树),以检查表达式的类型一致性。这可以说是最好的方法,因为它将解析与类型检查分开,使事情更清晰,并且通常允许更好的错误消息。
bool_expr
arithm_expr
直接确定 ID 的类型,并对布尔 ID 与算术 ID 使用不同的令牌。这可以通过使用符号表来完成,该符号表记录范围内的定义,并根据声明返回不同的标记。它通常要求声明始终在使用之前发生,但许多语言仍然要求这样做。
评论