主要思想是利用该语法自底向上地构造出1——>999999999999;
然后将数字转换成中文大写。
利用flex进行词法分析,bison进行句法分析。
语法如下:
e1-->one|two|three|four|five|six|seven|eight|nine
e2-->ten|eleven|twelve|thirteen|fourteen|fifteen|sixteen|seventeen|eighteen|nineteen
e3-->twenty|thirty|forty|fifty|sixty|seventy|eighty|ninety
e4-->e3-e1
|e3
|e2
|e1
e51-->e1 hundred and e4
|e1 hundred
e52-->e51
|e4
e61-->e52 thousand e51
|e52 thousand and e4
|e52 thousand
e62-->e61
|e52
e71-->e52 million e61
|e52 million e51
|e52 million and e4
|e52 million
e72-->e71
|e62
e81-->e52 billion e71
|e52 billion e61
|e52 billion e51
|e52 billion and e4
|e52 billion
e82-->e81
|e72
|zero
lex.l文件如下:
%% zero {yylval=0;return ZERO;} one {yylval=1;return NUM1;} two {yylval=2;return NUM1;} three {yylval=3;return NUM1;} four {yylval=4;return NUM1;} five {yylval=5;return NUM1;} six {yylval=6;return NUM1;} seven {yylval=7;return NUM1;} eight {yylval=8;return NUM1;} nine {yylval=9;return NUM1;} ten {yylval=10;return NUM2;} eleven {yylval=11;return NUM2;} twelve {yylval=12;return NUM2;} thirteen {yylval=13;return NUM2;} fourteen {yylval=14;return NUM2;} fifteen {yylval=15;return NUM2;} sixteen {yylval=16;return NUM2;} seventeen {yylval=17;return NUM2;} eighteen {yylval=18;return NUM2;} nineteen {yylval=19;return NUM2;} twenty {yylval=20;return NUM3;} thirty {yylval=30;return NUM3;} forty {yylval=40;return NUM3;} fifty {yylval=50;return NUM3;} sixty {yylval=60;return NUM3;} seventy {yylval=70;return NUM3;} eighty {yylval=80;return NUM3;} ninety {yylval=90;return NUM3;} hundred {return HUNDRED;} thousand {return THOUSAND;} million {return MILLION;} billion {return BILLION;} and {return AND;} [ ]+ /* skip blanks */ .| { return yytext[0];}
parser.y文件如下:
%{ #include<cmath> #include <string> #include <iostream> using namespace std; void yyerror(string msg); string trans(int term); void dtos(long term,string &result); int yylex(); #define YYSTYPE long %} %token NUM1 %token NUM2 %token NUM3 %token ZERO %token HUNDRED %token THOUSAND %token MILLION %token BILLION %token AND %% lines : lines e82 ' ' {string result;dtos($2,result);cout<<result<<endl;} | lines ' ' | /*empty*/ |error ' ' {yyerrok;} ; e82:e81 {$$=$1;} |e72 {$$=$1;} |ZERO {$$=0;} ; e81:e52 BILLION e71 {$$=$1*pow(10,9)+$3;} |e52 BILLION e61 {$$=$1*pow(10,9)+$3;} |e52 BILLION e51 {$$=$1*pow(10,9)+$3;} |e52 BILLION AND e4 {$$=$1*pow(10,9)+$4;} |e52 BILLION {$$=$1*pow(10,9);} ; e72:e71 {$$=$1;} |e62 {$$=$1;} ; e71:e52 MILLION e61 {$$=$1*pow(10,6)+$3;} |e52 MILLION e51 {$$=$1*pow(10,6)+$3;} |e52 MILLION AND e4 {$$=$1*pow(10,6)+$4;} |e52 MILLION {$$=$1*pow(10,6);} ; e62:e61 {$$=$1;} |e52 {$$=$1;} ; e61:e52 THOUSAND e51 {$$=$1*pow(10,3)+$3;} |e52 THOUSAND AND e4 {$$=$1*pow(10,3)+$4;} |e52 THOUSAND {$$=$1*pow(10,3);} ; e52:e51 {$$=$1;} |e4 {$$=$1;} ; e51:e1 HUNDRED AND e4 {$$=$1*pow(10,2)+$4;} |e1 HUNDRED {$$=$1*pow(10,2);} ; e4:e3'-'e1{$$=$1+$3;} |e3 {$$=$1;} |e2 {$$=$1;} |e1 {$$=$1;} ; e3:NUM3 {$$=$1;} ; e2:NUM2 {$$=$1;} ; e1:NUM1 {$$=$1;} ; %% #include "lex.yy.c" int main() { return yyparse(); } void yyerror(string msg) { cout<<msg<<endl; } string trans(int term) { if(term==1) return "壹"; if(term==2) return "贰"; if(term==3) return "叁"; if(term==4) return "肆"; if(term==5) return "伍"; if(term==6) return "陆"; if(term==7) return "柒"; if(term==8) return "捌"; if(term==9) return "玖"; } void dtos(long term,string &result) { bool flag1=false,flag2=false,flag3=false,flag4=false; long temp1=term/100000000; if(temp1!=0) { flag1=true; dtos(temp1,result); result+="亿"; } term%=100000000; long temp2=term/10000; if(temp2!=0) { if(temp2<1000) { if(flag1) result+="零"; } flag2=true; dtos(temp2,result); result+="万"; } else if(flag1) { result+="零";} term%=10000; long temp3=term/1000; if(temp3!=0) { flag3=true; result+=trans(temp3); result+="仟"; } else if(flag2) {result+="零";} term%=1000; long temp4=term/100; if(temp4!=0) { flag4=true; result+=trans(temp4); result+="佰"; } else if(flag3) {result+="零";} term%=100; long temp5=term/10; if(temp5!=0) { result+=trans(temp5); result+="拾"; } else if(flag4) {result+="零";} term%=10; long temp6=term; if(temp6!=0) result+=trans(temp6); if(result=="") result+="零"; string temp=result.substr(result.length()-3,result.length()); if((result!="零")&&(temp=="零")) { result=result.substr(0,result.length()-3); } }
程序的运行:
flex lex.l
bison parser.y
g++ parser.tab.c -ly -ll
./a.out
输入英文数字回车得到对应的中文大写。
版权声明:本文为博主原创文章,未经博主允许不得转载。