将yyin改为argv [1] Flex&野牛

时间:2016-11-15 08:41:17

标签: c++ bison lex argv

我正在研究Flex&野牛项目。我得到了我的flex&野牛完美的工作,但我想把argv作为输入(yyin)。所以我改变了yyin以便它采用了argv [1],但它实际上不起作用。似乎它花了argv [1]但是后来我得到了语法错误,即使我的字符串我想要完美地工作。

这是我的flex:

%{
#include "parser.hpp"
extern int yyparse();
%}

%option noyywrap


texte  [a-zA-z]+
entier [0-9]+(\.[0-9])?

%%

{entier} { yylval.num = atoi(yytext); return(NUMBER);}

"pi" return(PI);
"," return(SEP);
"(" return(OP);
")" return(CP);
"+" return(ADD);
"-" return(SUB);
"*" return(MUL);
"/" return(DIV);
"%" return (MODULO);
"sin" return(SIN);
"cos" return(COS);
"tan" return(TAN);
"acos" return(ACOS);
"asin" return(ASIN);
"atan" return(ATAN);
"sqrt" return(ROOT);
"pow" return(POW);
"exp" return(EXP);
"\n" return(END);
{texte} return(ERROR);
%%

然后我的野牛(我没有实现COS SIN和其他最简单的阅读):

 %defines
%{
#include <iostream>
#include <math.h>
using namespace std;
extern int yylex();
extern void yyerror(char const* msg);
%}

%union {double num;}

/* CARACTERES */
%token <num> NUMBER PI
%token OP CP SEP END

/* TRIGO */
%token COS SIN TAN
%token ACOS ASIN ATAN
/* CALCUL */
%token ADD SUB
%token MUL DIV
%token ROOT POW MODULO EXP ABS

/* CALCUL ORDER */
%left ADD SUB
%left MUL DIV

/* TRASH */
%token ERROR

%type <num> calclist exp factor term
%start calclist
%%
calclist: /* nothing */
 | calclist exp END {cout << $2 << endl;}
 ;
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
 | OP exp CP {$$ = $2;}
 ;
%%



extern void yyerror(char const* msg){
  cerr << "Error " << msg << endl;
}

然后我的主要:

#include <iostream>
#include "parser.hpp"
#include <string.h>
using namespace std;

extern FILE *yyin;
extern int yy_scan_string(const char *);


int main(int argc, char const *argv[]) {
  /*string buf(argv[1]);
  buf.append("\0");*/
  yy_scan_string(argv[1]);
  return yyparse();
}

最后是我的makefile:

all: bison flex main.cpp
    g++ parser.cpp lexer.cpp main.cpp -o parser
    rm lexer.cpp parser.cpp parser.hpp
    ./parser "(1+2)"
bison: parser.y
    bison -o parser.cpp parser.y
flex: lexer.l
    flex -o lexer.cpp lexer.l

我也试过了./parser (1+2)但是我得到了更多的错误。 谢谢你的帮助!

2 个答案:

答案 0 :(得分:1)

Flex & Bison book(第124页):

  

例程yy_scan_bytesyy_scan_string 创建一个缓冲区,其中包含要扫描的文本的副本。

(强调我的)

  

创建字符串缓冲区后,使用yy_switch_to_buffer告诉扫描程序从中读取...

yy_scan_string 创建您需要明确使用的缓冲区对象:

YY_BUFFER_STATE bp = yy_scan_string(...);  // Creates a buffer from the string
yy_switch_to_buffer(bp);                   // Use the buffer
int ret = yyparse();                       // Parse the string
yy_delete_buffer(bp);                      // Free the buffer

答案 1 :(得分:0)

我终于找到了答案。 我juste不得不改变我的主要内容:

string str = argv[1];
str += '\n';
yy_scan_string(str.c_str());

因为yyin是0-Terminated。 希望这会帮助其他人!