C 野牛的冲突
我试图使用Bison和Flex制作一个小型编译器,但我不断地变换/减少错误。我怀疑问题在于“转让”规则和“布尔诺”、“比较”、“表达”、“终结”、“效力”规则。 可能语法定义不正确。 我希望通过使用“%prec”来解决这个问题,但没有成功 然后我把代码放在Bison和errors上:C 野牛的冲突,c,compiler-construction,bison,lex,C,Compiler Construction,Bison,Lex,我试图使用Bison和Flex制作一个小型编译器,但我不断地变换/减少错误。我怀疑问题在于“转让”规则和“布尔诺”、“比较”、“表达”、“终结”、“效力”规则。 可能语法定义不正确。 我希望通过使用“%prec”来解决这个问题,但没有成功 然后我把代码放在Bison和errors上: ... %start programa %token <id> IDENTIFICADOR %token <intvalue> ENTERO %token <floatvalue>
...
%start programa
%token <id> IDENTIFICADOR
%token <intvalue> ENTERO
%token <floatvalue> FLOTANTE
%token <boolvalue> BOOLEANO
%token <stringvalue> CADENA
%token <etiqueta> IF ELSEIF ELSE FOR WHILE DO IN READ WRITE PRINT BREAK CONTINUE PASS EXCEPT TO DOWNTO
%token NL EOF_INS APAR OPAR ACLAU OCLAU
%left OR
%left AND
%nonassoc PEQUENO GRANDE PEQIG GRAIG IGUAL DIFFE
%left MENOS MAS
%left DIVISION_ENTERA DIVISION PRODUCTO MODULO
%right EXPONENTE
%right ASSIGNACIONS
%nonassoc IFX
%nonassoc ELSE
%%
programa : bloques;
bloques : bloque bloques
| bloque
;
bloque : if_sentencia
| while_sentencia
| for_sentencia
| do_sentencia
| instruccion
| NL
;
if_sentencia : IF expresion ACLAU bloques OCLAU %prec IFX
| IF expresion ACLAU bloques OCLAU else_sentencia
;
else_sentencia : ELSE expresion ACLAU bloques OCLAU
| ELSEIF expresion ACLAU bloques OCLAU else_sentencia
;
while_sentencia: WHILE expresion ACLAU bloques OCLAU
;
for_sentencia: FOR APAR assignacion EOF_INS expresion EOF_INS assignacion OPAR ACLAU bloques OCLAU
;
do_sentencia: DO ACLAU bloques OCLAU WHILE expresion EOF_INS
;
instruccion : expresion EOF_INS;
expresion: expresion OR expresion
| booleano;
booleano: booleano AND booleano
| comparacion;
comparacion: comparacion PEQUENO comparacion
| comparacion GRANDE comparacion
| comparacion PEQIG comparacion
| comparacion GRAIG comparacion
| comparacion IGUAL comparacion
| comparacion DIFFE comparacion
| expresionArit
;
expresionArit: expresionArit MAS expresionArit
| expresionArit MENOS expresionArit
| termino;
termino: termino PRODUCTO termino
| termino DIVISION termino
| termino DIVISION_ENTERA termino
| termino MODULO termino
| potencia
;
potencia: potencia EXPONENTE potencia
| factor
;
factor: ENTERO {printd("INT\n");}
| FLOTANTE {printd("FLOAT\n");}
| BOOLEANO {printd("BOOL\n");}
| CADENA {printd("STRING\n");}
| APAR expresion OPAR
| assignacion
;
assignacion: variable ASSIGNACIONS expresion {printd("ASSIGNACION");} | variable;
// a=b=c;
variable: IDENTIFICADOR {printd("VARIABLE\n");};
%%
int
yyerror (char const *s)
{
printf ("%s\n", s);
}
void main(int argc, char *argv[])
{
extern FILE *yyin;
++argv; --argc;
yyin = fopen( argv[0], "r" );
yyparse();
}
我查看了输出结果,有一些产品产生了错误,但没有解决它们
estado 21
19 expresion: booleano .
20 booleano: booleano . AND booleano
AND desplazar e ir al estado 38
AND [reduce usando la regla 19 (expresion)]
$default reduce usando la regla 19 (expresion)
estado 22
21 booleano: comparacion .
22 comparacion: comparacion . PEQUENO comparacion
23 | comparacion . GRANDE comparacion
24 | comparacion . PEQIG comparacion
25 | comparacion . GRAIG comparacion
26 | comparacion . IGUAL comparacion
27 | comparacion . DIFFE comparacion
DIFFE desplazar e ir al estado 39
IGUAL desplazar e ir al estado 40
GRAIG desplazar e ir al estado 41
PEQIG desplazar e ir al estado 42
GRANDE desplazar e ir al estado 43
PEQUENO desplazar e ir al estado 44
DIFFE [reduce usando la regla 21 (booleano)]
IGUAL [reduce usando la regla 21 (booleano)]
GRAIG [reduce usando la regla 21 (booleano)]
PEQIG [reduce usando la regla 21 (booleano)]
GRANDE [reduce usando la regla 21 (booleano)]
PEQUENO [reduce usando la regla 21 (booleano)]
$default reduce usando la regla 21 (booleano)
estado 23
28 comparacion: expresionArit .
29 expresionArit: expresionArit . MAS expresionArit
30 | expresionArit . MENOS expresionArit
MAS desplazar e ir al estado 45
MENOS desplazar e ir al estado 46
MAS [reduce usando la regla 28 (comparacion)]
MENOS [reduce usando la regla 28 (comparacion)]
$default reduce usando la regla 28 (comparacion)
estado 24
31 expresionArit: termino .
32 termino: termino . PRODUCTO termino
33 | termino . DIVISION termino
34 | termino . DIVISION_ENTERA termino
35 | termino . MODULO termino
MODULO desplazar e ir al estado 47
PRODUCTO desplazar e ir al estado 48
DIVISION desplazar e ir al estado 49
DIVISION_ENTERA desplazar e ir al estado 50
MODULO [reduce usando la regla 31 (expresionArit)]
PRODUCTO [reduce usando la regla 31 (expresionArit)]
DIVISION [reduce usando la regla 31 (expresionArit)]
DIVISION_ENTERA [reduce usando la regla 31 (expresionArit)]
$default reduce usando la regla 31 (expresionArit)
estado 25
36 termino: potencia .
37 potencia: potencia . EXPONENTE potencia
EXPONENTE desplazar e ir al estado 51
EXPONENTE [reduce usando la regla 36 (termino)]
$default reduce usando la regla 36 (termino)
表达式语法
算术表达式有两种常见的语法书写方式:
1.具有优先声明。
在这种情况下,我们指定每个运算符的优先级,从最低优先级开始(通常为+和−). 优先级越高,运算符与其操作数的绑定越紧密。例如,在3+4*5
中,*优先于+,因为表达式被解析为3+(4*5)
)
在本例中,只有一个“表达式”非终结符,我们使用该非终结符编写所有规则:
%left '+' '-'
%left '*' '/' '%'
/* ... */
%nonassoc '<' "<=" "==" "!=" ">=" '>'
%left "&&"
%left "||"
%%
expression: expression "||" expression
| expression "&&" expression
| expression "<=" expression
| expression '+' expression
| expression '-' expression
/* etcetera */
| NUMBER
| IDENTIFIER
| '(' expression ')'
%左'+''-'
%左'*''/''%
/* ... */
%非ASSOC“
%左“&&”
%左“| |”
%%
表达式:表达式“| |”表达式
|表达式“&&”表达式
|表情“编写表达式语法时,您应该在标记上使用优先级,并在表达式中使用所有规则,或者使用多个规则来表示优先级。你不能真的混搭,一部分用一种方法,另一部分用另一种方法。这里您尝试使用%left
/%right
设置左/右关联,并使用多个规则设置优先级。你需要决定其中一个
要仅使用优先级声明,请将所有非端子booleano
、comparison
等替换为expression
。然后删除冗余的expression:expression
规则,因为这些规则会导致冲突
若要使用规则作为优先级,请将运算符前后具有相同非终结符的规则更改为在一侧使用下一级别的非终结符来设置左关联性或右关联性:
expresion: expresion OR booleano /* left associative */
comparacion: expresionArit PEQUENO expresionArit /* non-associative */
potencia: factor EXPONENTE potencia /* right associative */
如果我用表达式和rici说的话来替换。我完全明白!我感谢你的帮助!
expr : expr0
expr0: expr0 '+' expr1 /* Left associative */
| expr0 '-' expr1
| expr1 /* There is always a unit rule */
expr1: expr1 '*' expr2
| expr1 '/' expr2
| expr1 '%' expr2
| expr2
/* ... */
expr3: expr4 "**" expr3 /* Right associative */
| expr4
expr4: expr5 "<=" expr5 /* Non-associative */
| expr5 "==" expr5
| expr5 "!=" expr5
| expr5 ">=" expr5
| expr5 '<' expr5
| expr5 '>' expr5
| expr5
expr5: expr5 "&&" expr6
| expr6
expr6: expr6 "||" expr7
| expr7
expr7: '-' expr7 /* Prefix operator */
| expr8
expr8: expr8 '[' expr ']' /* Postfix operator */
| term
term : NUMBER | IDENTIFIER | '(' expr ')'
expresion: expresion OR booleano /* left associative */
comparacion: expresionArit PEQUENO expresionArit /* non-associative */
potencia: factor EXPONENTE potencia /* right associative */