2016-11-15 107 views
4

我正在開發Flex & Bison項目。我得到了我的flex &野牛完美的工作,但我試圖給argv作爲輸入(yyin)。所以我改變了yyin,以至於花了argv [1],但它實際上並不工作。似乎它採用了argv [1],但是即使我的字符串我想完美工作,我也得到了一個語法錯誤。將yyin更改爲argv [1] Flex&Bison

這裏是我的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(); 
} 

最後我的生成文件:

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),但我得到了更多的錯誤。 感謝您的幫助!

回答

1

Flex & Bison book(第124頁):

例程yy_scan_bytesyy_scan_string創建緩衝區與文本的副本進行掃描。

(重點煤礦)

而且

一旦創建一個字符串緩衝區,使用yy_switch_to_buffer告訴掃描器從中讀取...

yy_scan_string只有創建一個緩衝區對象,您需要明確地u se:

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 
+0

非常感謝!我是否需要爲YY_BUFFER_STATE添加一些內容?我得到了一個錯誤,並切換到緩衝區! –

+0

@ G.Courmont我不記得了,那是很久以前我自己用過Flex/Bison的。我現在基本上只是閱讀鏈接的書,在我閱讀的章節中沒有提到任何內容。您必須搜索自動生成的頭文件以及與系統上安裝的Flex/Bison相關的其他頭文件。 –

+0

這個答案不正確。請參閱flex文檔。 – rici

0

我終於找到答案了。 我中庸之道不得不改變一點點我的主要:

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

因爲yyin中是0終結。 希望這會幫助別人!