实数的flex和bison问题

时间:2017-05-11 22:33:30

标签: flex-lexer

我一直在使用flex和bison制作一个小型计算器。我的文件如下:

bisonFile.y

%{
#include <stdio.h>
%}
/* declare tokens */
%token NUMBER
%token ADD SUB MUL DIV ABS
%token EOL
%%
calclist: /* nothing */
| calclist exp EOL { printf("= %d\n", $2); }
;
exp: factor
| exp ADD factor { $$ = $1 + $3; }
| exp SUB factor { $$ = $1 - $3; }
;
factor: term
| factor MUL term { $$ = $1 * $3; }
| factor DIV term { $$ = $1 / $3; }
;
term: NUMBER
| ABS term { $$ = $2 >= 0? $2 : - $2; }
;
%%
main(int argc, char **argv)
{
yyparse();
}
yyerror(char *s)
{
fprintf(stderr, "error: %s\n", s);
}

flexFile.l

%{
# include "f5.tab.h"
int yylval;
%}
/* reconocimiento de tokens e impresion */
%{
int yylval;
%}
%option noyywrap
%%
"+" { return ADD; }
"-" { return SUB; }
"*" { return MUL; }
"/" { return DIV; }
"|" { return ABS; }
[0-9]+("."[0-9]+)? { yylval = atoi(yytext); return NUMBER; }   //part added
\n { return EOL; }
[ \t] { /* ignore whitespace */ }
. { printf("Mystery character %c\n", *yytext); }
%%

我的程序适用于整数,它也可以识别实数,但问题是当我打印一个操作的结果时,它总是将答案作为整数返回。那是为什么?

由于

2 个答案:

答案 0 :(得分:0)

您在生产中使用atoi会将字符串转换为整数。 使用atof会将其转换为浮点数。 如果要将两者分开,则需要更改整数的匹配规则,并为浮点添加一个。

答案 1 :(得分:0)

在“bisonFile.y”文件中更改"%d""%f"。这使用浮点格式来打印结果。固定行应为:

| calclist exp EOL { printf("= %f\n", $2); }

在“flexFile.l”文件中,删除两个定义int yylval。野牛输出

YYSTYPE yylval;

自动。 YYSTYPE是语义值的类型。因为你想要一个浮点计算器,所以它应该是double。请注意,YYSTYPE默认为int。要更改它,在编译C代码时必须定义YYSTYPE(来自bison和flex)(见下文)。

最后,正如MIS已经说明的那样,替换atoi()atof()。 flexFile.l中编辑的行应为:

[0-9]+("."[0-9]+)? { yylval = atof(yytext); return NUMBER; }

对于初学者来说,flex和bison源之间的依赖关系可能会令人困惑。最小的Makefile记录了如何编译示例。第2行始终如一地设置扫描程序和解析器的语义类型:

calc:               calc.o l.o
calc.o l.o:         CFLAGS+=-DYYSTYPE=double
l.o:                l.c f5.tab.h
calc.c f5.tab.h:    bisonFile.y
                    bison -o $@ --defines=f5.tab.h $^
l.c:                flexFile.l f5.tab.h
                    flex -o $@ $^
clean::
                    $(RM) calc calc.o calc.c f5.tab.h l.o l.c

那就行了。