3

我正在阅读 LFS 并遇到了我在其他地方之前见过的野牛,所以我想我应该更多地了解它。我从 UC Riverside CS 部门找到了这个页面,示例代码不起作用。有谁知道怎么了?为方便起见,我粘贴了代码:

calc.lex文件:

/* Mini Calculator */
/* calc.lex */

%{
#include "heading.h"
#include "tok.h"
int yyerror(char *s);
int yylineno = 1;
%}

digit       [0-9]
int_const   {digit}+

%%

{int_const} { yylval.int_val = atoi(yytext); return INTEGER_LITERAL; }
"+"     { yylval.op_val = new std::string(yytext); return PLUS; }
"*"     { yylval.op_val = new std::string(yytext); return MULT; }

[ \t]*      {}
[\n]        { yylineno++;   }

.       { std::cerr << "SCANNER "; yyerror(""); exit(1);    }

calc.y文件:

/* Mini Calculator */
/* calc.y */

%{
#include "heading.h"
int yyerror(char *s);
int yylex(void);
%}

%union{
  int       int_val;
  string*   op_val;
}

%start  input 

%token  <int_val>   INTEGER_LITERAL
%type   <int_val>   exp
%left   PLUS
%left   MULT

%%

input:      /* empty */
        | exp   { cout << "Result: " << $1 << endl; }
        ;

exp:        INTEGER_LITERAL { $$ = $1; }
        | exp PLUS exp  { $$ = $1 + $3; }
        | exp MULT exp  { $$ = $1 * $3; }
        ;

%%

int yyerror(string s)
{
  extern int yylineno;  // defined and maintained in lex.c
  extern char *yytext;  // defined and maintained in lex.c

  cerr << "ERROR: " << s << " at symbol \"" << yytext;
  cerr << "\" on line " << yylineno << endl;
  exit(1);
}

int yyerror(char *s)
{
  return yyerror(string(s));
}

这是错误消息:

$ make
bison -d -v calc.y
cp calc.tab.c bison.c
cmp -s calc.tab.h tok.h || cp calc.tab.h tok.h
g++ -g -Wall -ansi -pedantic -c bison.c -o bison.o
calc.tab.c: In function ‘int yyparse()’:
calc.tab.c:1381: warning: deprecated conversion from string constant to ‘char*’
calc.tab.c:1524: warning: deprecated conversion from string constant to ‘char*’
flex calc.lex
cp lex.yy.c lex.c
g++ -g -Wall -ansi -pedantic -c lex.c -o lex.o
calc.lex:8: error: redefinition of ‘int yylineno’
lex.yy.c:349: error: ‘int yylineno’ previously defined here
calc.lex: In function ‘int yylex()’:
calc.lex:23: warning: deprecated conversion from string constant to ‘char*’
lex.yy.c: At global scope:
lex.yy.c:1105: warning: ‘void yyunput(int, char*)’ defined but not used
make: *** [lex.o] Error 1
4

1 回答 1

3

问题是您正在使用 C++ 编译器编译 C 代码。如果你想让 flex 生成 C++,有一个命令行选项。

扫描仪生成的代码yylineno已经提供了一个定义。

在 C 中,以下是可以接受的:

int yylineno;         /* tentative definition */
int yylineno = 1;     /* definition */

在 C++ 中,它不是:

int yylineno;         /* definition */
int yylineno = 1;     /* another definition: duplicate, error! */

另请注意,-ansigcc 选项适用于 C 方言。

关于字符串常量的警告也是因为 C++。在 C++ 中,类似字符串文字的"abc"计算结果为 a const char *,而不是char *.

最后,请注意 flex 生成的扫描器不会自动包含更新yylineno. 使用%option yylineno. 查看 Flex 上的 GNU 信息手册。

于 2013-10-08T19:18:40.113 回答