【发布时间】:2018-07-02 21:14:58
【问题描述】:
我正在编写类似 PHP 的解释器,但在 shift/reduce 和 reduce/reduce 冲突方面存在一些问题。 有人可以帮我理解 shift/reduce 和 reduce/reduce 冲突。
我必须编写和解释发送/回显非值并评估以“魔术”@ 字符开头的表达式,例如@if(cond) ... @end;。所以 "if" 必须被回显,而 @if(cond) 应该被解释
问题:scriptlang.y 包含 21 个 shift/reduce 冲突和 2 个 reduce/reduce 冲突。
%union {
char* sval;
}
%token <sval> IDENTIFIER
%token <sval> RBRACKET
%token <sval> LBRACKET
%token <sval> KWSWITCH
%token <sval> KWIF
%token <sval> MAGICESC
%token MAGIC
%token ENDSTM
%type <sval> filechar
%start script
%%
script:
commands
;
commands:
/* empty */
| command
| commands command
;
command:
filechar {
analyser_echo($1,"filechar",analyser_canEcho);
}
| magic_command {}
;
filechar:
IDENTIFIER
| LBRACKET
| RBRACKET
| KWSWITCH
| KWIF
| MAGICESC
;
magic_command:
MAGIC valuation
| MAGIC alternative
;
valuation:
LBRACKET IDENTIFIER RBRACKET {
fprintf(yyout, "<val>");
}
;
alternative:
switch_alternative
| if_alternative
;
switch_alternative:
switch_block end_stm
;
switch_block:
switch_stm
| switch_stm commands
;
switch_stm:
KWSWITCH LBRACKET IDENTIFIER RBRACKET {}
;
if_alternative:
if_block end_stm
;
if_block:
if_stm
| if_stm commands
;
if_stm:
KWIF LBRACKET IDENTIFIER RBRACKET {}
;
end_stm:
ENDSTM
;
%%
flex 文件内容:
"@@" {
yylval.sval = "@";
return MAGICESC;
}
"@" {
return MAGIC;
}
"(" {
yylval.sval = yytext;
return LBRACKET;
}
")" {
yylval.sval = yytext;
return RBRACKET;
}
"@end;" {
return ENDSTM;
}
"if" {
yylval.sval = yytext;
return KWIF;
}
"switch" {
yylval.sval = yytext;
return KWSWITCH;
}
[a-zA-Z][_a-zA-Z0-9]* {
yylval.sval = yytext;
return IDENTIFIER;
}
\n|. {
if(analyser_canEcho>0){
ECHO;
}
}
%%
【问题讨论】:
标签: bison flex-lexer