【发布时间】:2015-04-14 23:56:34
【问题描述】:
当解析器试图找出某事物是一元运算符还是二元运算符时,我不断地遇到移位/减少冲突。
%token <intconst> tHEX tOCT tDEC tRUNE
%token <stringconst> tBOOL INTERPRETEDSTRING RAWSTRING tIDENTIFIER
%token <floatconst> tFLOAT
%token <charconst> tRUNES
%token TRUE FALSE BREAK CASE CHAN CONST CONTINUE DEFAULT DEFER ELSE FALLTHROUGH FOR FUNC GO GOTO IF IMPORT INTERFACE MAP PACKAGE RANGE RETURN SELECT STRUCT SWITCH TYPE VAR INT PRINT FLOAT PRINTLN BOOL APPEND RUNE STRING SEMICOLON NEWLINE PLUS MINUS TIMES DIV MOD AMP PIPE CARAT COUT CIN AMPCARAT SELFPLUS SELFMINUS SELFTIMES SELFDIV SELFMOD AMPEQUALS PIPEEQUALS CARATEQUALS COUTEQUALS CINEQUALS WTF AND OR REDIRECT INCREMENT DECREMENT DOESEQUALS LT GT EQUALS NOT NEQ LE GE COMPAT ELLIPSIS LEFTPAREN RIGHTPAREN LEFTSQUARE RIGHTSQUARE LEFTBRACE RIGHTBRACE COMMA PERIOD FULLCOLON ESCAPEA ESCAPEB ESCAPEF ESCAPEV ESCAPESLASH ESCAPEAPOSTROPHE INVALID
/*%token unary*/
/*%token binary*/
%left OR
%left AND
%left DOESEQUALS NEQ GT GE LT LE
%left PLUS MINUS PIPE CARAT
%left TIMES DIV MOD COUT CIN AMP AMPCARAT
/*%left binary*/
%left UPLUS UMINUS UNOT UCARAT UTIMES UAMP UPAREN
%start expList
%%
expList: exp expList {}
| /*empty*/
;
exp: exp OR addOp {}
| exp AND addOp {}
| exp NEQ addOp {}
| exp GT addOp {}
| exp GE addOp {}
| exp LT addOp {}
| exp LE addOp {}
| addOp {}
;
addOp: addOp PLUS mulOp {}
| addOp MINUS mulOp {}
| addOp PIPE mulOp {}
| addOp CARAT mulOp {}
| mulOp {}
;
mulOp: mulOp TIMES factor {}
| mulOp DIV factor {}
| mulOp MOD factor {}
| mulOp COUT factor {}
| mulOp CIN factor {}
| mulOp AMP factor {}
| mulOp AMPCARAT factor {}
| factor {}
;
factor: LEFTPAREN exp RIGHTPAREN %prec UPAREN {}
| PLUS factor %prec UPLUS {}
| MINUS factor %prec UMINUS {}
| NOT factor %prec UNOT {}
| CARAT factor %prec UCARAT {}
| TIMES factor %prec UTIMES {}
| AMP factor %prec UAMP {}
| tIDENTIFIER {}
| tDEC {}
| tFLOAT {}
| tOCT {}
| tHEX {}
| tRUNES {}
| INTERPRETEDSTRING {}
| RAWSTRING {}
;
%%
我知道我有很多代币,我最终会使用它们。我只是想让表达式工作的语法。 这是我遇到的移位/减少错误
State 18
10 exp: addOp .
11 addOp: addOp . PLUS mulOp
12 | addOp . MINUS mulOp
13 | addOp . PIPE mulOp
14 | addOp . CARAT mulOp
PLUS shift, and go to state 37
MINUS shift, and go to state 38
PIPE shift, and go to state 39
CARAT shift, and go to state 40
PLUS [reduce using rule 10 (exp)]
MINUS [reduce using rule 10 (exp)]
CARAT [reduce using rule 10 (exp)]
$default reduce using rule 10 (exp)
State 19
15 addOp: mulOp .
16 mulOp: mulOp . TIMES factor
17 | mulOp . DIV factor
18 | mulOp . MOD factor
19 | mulOp . COUT factor
20 | mulOp . CIN factor
21 | mulOp . AMP factor
22 | mulOp . AMPCARAT factor
TIMES shift, and go to state 41
DIV shift, and go to state 42
MOD shift, and go to state 43
AMP shift, and go to state 44
COUT shift, and go to state 45
CIN shift, and go to state 46
AMPCARAT shift, and go to state 47
TIMES [reduce using rule 15 (addOp)]
AMP [reduce using rule 15 (addOp)]
$default reduce using rule 15 (addOp)
我的想法已经用完了,如果我能得到任何帮助,我将不胜感激。
【问题讨论】:
-
expList有分隔符吗?很难判断表达式 5 - 5 是否是其中的一个或两个表达式... -
我不确定你所说的分隔符是什么意思。使用表达式列表,使读入的一系列表达式有效。
-
让我重新表述一下这个问题:解析器如何确定表达式结尾?
-
那么 5 - 5 序列是一个表达式(减法)还是两个:5 和 - 5?
-
顺便说一句,所有这些优先级声明都没有做任何事情。因此,您不妨将它们全部摆脱;它们只会使您的语法变得不必要地复杂和难以阅读。如果您打算使用优先级声明,您将只有一个表达式非终结符(而不是四个)。在这种情况下,您仍然只需要一个用于一元表达式的伪标记,因为所有一元产生式都具有相同的优先级,并且您不需要为带括号的表达式声明优先级,因为该产生式不能参与移位减少冲突。
标签: c bison yacc operator-precedence shift-reduce-conflict