Yacc / Bison yyerror

时间:2015-11-18 19:18:49

标签: parsing bison yacc

我正在尝试创建自己的错误,但显然,调用yyerror()并不足以告诉解析器存在错误。我举了一个例子来更好地描述我的问题。 所以这里有一个解析器,必须检查一个语句是否是两个带逗号的数字。数字不能以0开头。

输入yacc:

%token DIGIT            

%{      
#include <stdio.h> 
#include <stdlib.h>
void yyerror(char *s);    
%}

%%
    list:    |
        list stat ';' {printf("The statement is correct!\n\n");} |
        list error ';' {printf("The statement is incorrect!\n\n");}

    stat:   number ',' number

    number: DIGIT {if ($1==0) yyerror("number starts with 0");} | 
        number DIGIT {$$ = $1*10+$2;}
%%
extern int linenum;
void yyerror(char *s) {
    fprintf(stderr, " line %d: %s\n", linenum, s);
}

for lex:

%{
#include <stdio.h>
#include "y.tab.h"
int linenum = 1;
%} 

%% 

[0-9]               {
                 yylval = yytext[0] - '0';
                 return DIGIT;
                }

[ \t\r]+            ;
\n                  ++linenum;
.               return(yytext[0]);

解析器的输入:

34, 43;
32,fs;
03, 23;

这是输出:

The statement is correct!

 line 2: syntax error
The statement is incorrect!

 line 3: number starts with 0
The statement is correct!

即使找到了第3行的错误,解析仍然会继续。我该如何解决?

Upd:使用YYERROR解决了问题;

3 个答案:

答案 0 :(得分:0)

如果你想在检测到一个错误后停止(为什么?),只需从相关生产中返回。

默认情况下,它会执行错误恢复。

答案 1 :(得分:0)

解析正在继续,因为您的规则中包含error,这是一个错误恢复规则,告诉解析器如何从错误中恢复并继续。如果您在错误后想要继续,请删除错误恢复规则。然后yyparse将在发生错误后立即返回(非零)。

答案 2 :(得分:-1)

似乎我yyerror()只打印错误消息,但没有在解析器中设置错误状态。可能你可以修改一下语法吗?

法:

0          {
             yylval = 0;
             return ZERO;
           }
[1-9]      {
             yylval = yytext[0] - '0';
             return DIGITNOZERO;
           }

YACC:

number: DIGITNOZERO | 
        number DIGITNOZERO  {$$ = $1*10+$2;} | 
        number ZERO {$$ = $1*10;}