Posted on 2010-09-20 15:29
Prayer 閱讀(791)
評論(0) 編輯 收藏 引用 所屬分類:
C/C++
這里有一個使用bison建立一個簡單的計算器的例子:
http://www.cs.berkeley.edu/~maratb/cs164/bison.html
使用bison和flex工具學習編譯原理,遠比單獨看書然后自己編寫一些程序生動的多。這樣你就不會在那些復雜的字符處理,正則表達式的處理上浪費精力,最后費盡心力,卻沒有結果,失去了學習的興趣。
我
這里有一個簡單的計算器的程序,可以實現加、減、乘、除運算,并支持括號的處理和26個字母作為變量。以前自己使用后綴表達式方式寫過一個這樣的程序,單
單中綴表達式改為后綴表達式就是幾百行的代碼,反正自己現在還是不知道怎么處理里面復雜的堆棧的(我用了STL的List實現)。
詞法處理文件calc.lex內容如下:
%{
/*
* 一個簡單計算器的Lex詞法文件
*/
#include <stdlib.h>
void yyerror(char*);
#include "calc.tab.h"
%}
%%
/* a-z為變量 */
[a-z] {
yylval = *yytext - 'a';
return VARIABLE;
}
/* 整數 */
[0-9]+ {
yylval = atoi(yytext);
return INTEGER;
}
/* 運算符 */
[-+()=/*\n] {return *yytext;}
/* 空白被忽略 */
[ \t] ;
/* 其他字符都是非法的 */
. yyerror("無效的輸入字符");
%%
int yywrap(void)
{
return 1;
}
詞法處理的目標就是區分出來每個成員到底是什么,這里有兩種 INTEGER和VARIABLE。只要區分出來各個成分詞法分析的任務就完成了。
語法處理文件calc.y內容如下:
%token INTEGER VARIABLE
%left '+' '-'
%left '*' '/'
%{
#include <stdio.h>
void yyerror(char*);
int yylex(void);
int sym[26];
%}
%%
program:
program statement '\n'
|
;
statement:
expr {printf("%d\n", $1);}
|VARIABLE '=' expr {sym[$1] = $3;}
;
expr:
INTEGER
|VARIABLE{$$ = sym[$1];}
|expr '+' expr {$$ = $1 + $3;}
|expr '-' expr {$$ = $1 - $3;}
|expr '*' expr {$$ = $1 * $3;}
|expr '/' expr {$$ = $1 / $3;}
|'('expr')' {$$ = $2;}
;
%%
void yyerror(char* s)
{
fprintf(stderr, "%s\n", s);
}
int main(void)
{
printf("A simple calculator.\n");
yyparse();
return 0;
}
語法分析文件的寫法就是將BNF表達式描述一下即可,規則隨著條目逐漸細化,變成了可以理解的內容。這里不用管如何實現這些語法的分析,只是需要告知如何構建這些語法。
編譯命令如下:
>bison -d calc.y
>flex calc.lex
>gcc calc.tab.c lex.yy.c -o calc