我正在尝试使用flex
和yacc
来解析'C'源代码。不幸的是,我收到错误“预期的标识符'或''''''''''''''''''''''''... ... ... ...
这是我的flex文件(称为mini.l):
%{
%}
digit [0-9]
letter [a-zA-Z]
number (digit)+
id (letter|_)(letter|digit|_)*
integer (int)
character (char)
comma [,]
%%
{integer} {return INT;}
{character} {return CHAR;}
{number} {return NUM;}
{id} {return IDENTIFIER;}
{comma} {return ',';}
[-+*/] {return *yytext;}
. {}
%%
main()
{
yylex();
}
相应的yacc文件(名为my_yacc.y)如下所示:
%{
#include <ctype.h>
#include <stdio.h>
/* #include "myhead.h" */
#include "mini.l"
#define YYSTYPE double
# undef fprintf
%}
%token INT
%token CHAR
%token IDENTIFIER
%token NUM
%token ','
%left '+' '-'
%left '*' '/'
%right UMINUS
%%
lines:lines expr '\n' {printf("%g\n",$2);}
|lines '\n'
|D
|
;
expr :expr '*' expr {$$=$1*$3;}
|expr '/' expr {$$=$1/$3;}
|expr '+' expr {$$=$1+$3;}
|expr '-' expr {$$=$1+$3;}
|'(' expr ')' {$$=$2;}
|'-' expr %prec UMINUS {$$=-$2;}
|IDENTIFIER {}
|NUM {}
;
T :INT {}
|CHAR {}
;
L :L ',' IDENTIFIER {}
|IDENTIFIER {}
;
D :T L {printf("T is %g, L is %g",$1,$2);}
;
%%
/*void yyerror (char *s)
{
fprintf (stderr, "%s\n", s);
}
*/
我正在使用命令编译生成的代码:
flex mini.l
yacc my_yacc.y
gcc y.tab.c -ly
答案 0 :(得分:1)
您看到的错误来自C编译器,是由于您对flex和yacc如何协同工作的误解造成的。当我通过与你相同的工具运行它时,我也会遇到以下错误,正如@KeithThompson所指出的那样:
In file included from my_yacc.y:5:0:
mini.l:1:1: error: expected identifier or '(' before '%' token
%{
^
mini.l:5:1: error: expected '=', ',', ';', 'asm' or '__attribute__' before' letter'
letter [a-zA-Z]
^
mini.l:12:11: error: expected identifier or '(' before '{' token
{integer} {return INT;}
^
... elided the rest ...
虽然您已按正确的顺序使用了flex
,yacc
和gcc
命令,但您已将文件mini.l
包含在您的野牛输入中。这是不正确的。您应该包含mini.l
中创建的flex的输出。该文件名为lex.yy.c
。还必须在yacc输入文件的末尾包含此 。 (这是因为如果你没有得到@flolo发现的错误)。如果您对yacc
文件进行了必要的更改,那么您将拥有以下内容:
%{
#include <ctype.h>
#include <stdio.h>
/* #include "myhead.h" */
/* #include "mini.l" */
#define YYSTYPE double
# undef fprintf
%}
%token INT
%token CHAR
%token IDENTIFIER
%token NUM
%token ','
%left '+' '-'
%left '*' '/'
%right UMINUS
%%
lines:lines expr '\n' {printf("%g\n",$2);}
|lines '\n'
|D
|
;
expr :expr '*' expr {$$=$1*$3;}
|expr '/' expr {$$=$1/$3;}
|expr '+' expr {$$=$1+$3;}
|expr '-' expr {$$=$1+$3;}
|'(' expr ')' {$$=$2;}
|'-' expr %prec UMINUS {$$=-$2;}
|IDENTIFIER {}
|NUM {}
;
T :INT {}
|CHAR {}
;
L :L ',' IDENTIFIER {}
|IDENTIFIER {}
;
D :T L {printf("T is %g, L is %g",$1,$2);}
;
%%
void yyerror (char *s)
{
fprintf (stderr, "%s\n", s);
}
#include "lex.yy.c"
如果您现在运行命令序列,您会发现它既编译又运行,并且还正确处理C语言输入。
您犯的错误是新用户对工具flex
和yacc
造成的常见错误。