您好,欢迎访问三七文档
当前位置:首页 > 商业/管理/HR > 销售管理 > C语言实现中缀、后缀、前缀表达式-相互转化并求值
1.问题描述(1)表达式求值问题表达式是数据运算的基本形式。人们的书写习惯是中缀式,如:11+22*(7-4)/3。中缀式的计算按运算符的优先级及括号优先的原则,相同级别从左到右进行计算。表达式还有后缀式(如:2274-*3/11+)和前缀式(如:+11/*22–743)。后缀表达式和前缀表达式中没有括号,给计算带来方便。如后缀式计算时按运算符出现的先后进行计算。本设计的主要任务是进行表达式形式的转换及不同形式的表达式计算。2.数据结构设计(1)表达式求值问题由于表达式中有字符与数字两种类型,故定义结点一个标志域data,标志结点存储的为字符data=2还是数字data=1,再寻找结点中对应的存储位置,读取数字域data1,字符域data2。而在前缀表达式时,存在表达式逆序,因表达式类型不统一,用栈逆序极不方便,选择构建双向链表,存储表达式。typedefstructNode//定义存储中缀表达式的结点类型{intdata;intdata1;chardata2;structNode*next;}Lnode;typedefstructNode2//定义存储前缀表达式的结点类型{intdata;intdata1;chardata2;structNode2*next;structNode2*prior;}Lnode2;3.运行、测试与分析(1)表达式求值问题(1)按提示输入中缀表达式,如图1.1所示。如输入中缀表达式不正确,提示输入有误,如图1.2,1.3所示。图1.1图1.2图1.3(2)选择表达式转换并求值方式。按“1”选择中缀表达式求值,如图1.4所示。图1.4(3)按“2”选择中缀表达式转变为后缀表达式并求值,如图1.5所示。图1.5(4)按“3”选择中缀表达式转变为前缀表达式并求值,如图1.6所示。图1.6附录:源代码(1)表达式求值问题#includestdio.h#includestdlib.h#defineMAXNUM100typedefstructNode//定义存储中缀表达式的结点类型{intdata;intdata1;chardata2;structNode*next;}Lnode;typedefstructNode2//定义存储前缀表达式的结点类型{intdata;intdata1;chardata2;structNode2*next;structNode2*prior;}Lnode2;typedefintselemtype1;//定义运算数栈的结点typedefstruct//定义运算数栈的类型{selemtype1*base;selemtype1*top;}sqstack1;voidInitStack1(sqstack1&s)//新建一个空运算数栈{s.base=(selemtype1*)malloc(MAXNUM*sizeof(selemtype1));s.top=s.base;if(!s.base)printf(出错:申请空间失败!\n);}voidPush1(sqstack1&s,selemtype1&e)//运算数栈,入栈:插入元素e为新的栈顶元素{if(s.top-s.base=MAXNUM)printf(出错:表达式过长!1\n);*s.top++=e;}voidGetTop1(sqstack1s,selemtype1&e)//运算数栈,用e返回栈顶元素{e=*(s.top-1);}voidPopopnd1(sqstack1&s,selemtype1&e)//运算数栈,退栈:删除栈顶元素,并用e返回其值{e=*--s.top;}intstackempy1(sqstack1s)//运算数栈,若为空栈返回1,否则返回0{if(s.top==s.base)return1;elsereturn0;}typedefcharselemtype2;//定义运算符栈的结点类型typedefstruct//定义运算符栈类型{selemtype2*base;selemtype2*top;}sqstack2;voidInitStack2(sqstack2&s)//新建一个空运算符栈{s.base=(selemtype2*)malloc(MAXNUM*sizeof(selemtype2));s.top=s.base;if(!s.base)printf(出错:申请空间失败!\n);}voidPush2(sqstack2&s,selemtype2&e)//运算符栈,入栈:插入元素e为新的栈顶元素{if(s.top-s.base=MAXNUM)printf(出错:表达式过长!2\n);*s.top++=e;}voidGetTop2(sqstack2s,selemtype2&e)//运算符栈,用e返回栈顶元素{e=*(s.top-1);}voidPopopnd2(sqstack2&s,selemtype2&e)//运算符栈,退栈:删除栈顶元素,并用e返回其值{e=*--s.top;}intstackempy2(sqstack2s)//运算符栈,若为空栈返回1,否则返回0{if(s.top==s.base)return1;elsereturn0;}voidpriority(charc,int&i)//确定运算符优先级{if(c=='*'||c=='/'||c=='%')i=2;elseif(c=='+'||c=='-')i=1;elsei=0;}intcompare(chara,charb)//比较栈顶元素运算符与外部运算符优先级大小,外部优先级大则返回1,反之返回0{intin,out;priority(a,in);priority(b,out);if(outin)return1;elsereturn0;}voidOperat(sqstack1&OPND,sqstack2&OPTR){intnum1,num2,num;charc;Popopnd1(OPND,num2);Popopnd1(OPND,num1);Popopnd2(OPTR,c);switch(c){case'+':num=num1+num2;break;case'-':num=num1-num2;break;case'*':num=num1*num2;break;case'/':num=num1/num2;break;case'%':num=num1%num2;break;}Push1(OPND,num);}voidOperatqianzhui(sqstack1&OPND,sqstack2&OPTR){intnum1,num2,num;charc;Popopnd1(OPND,num1);Popopnd1(OPND,num2);Popopnd2(OPTR,c);switch(c){case'+':num=num1+num2;break;case'-':num=num1-num2;break;case'*':num=num1*num2;break;case'/':num=num1/num2;break;case'%':num=num1%num2;break;}Push1(OPND,num);}voidhouzhuiqiuzhi(Lnode*p,int&e)//后缀表达式求值{sqstack1OPND;//运算数栈sqstack2OPTR;//运算符栈intn;charc;p=p-next;InitStack1(OPND);InitStack2(OPTR);while(p){switch(p-data){case1:n=p-data1;Push1(OPND,n);break;case2:c=p-data2;Push2(OPTR,c);Operat(OPND,OPTR);break;default:printf(结点有误);break;}p=p-next;}Popopnd1(OPND,n);e=n;}voidzhongzhui(Lnode*p)//中缀表达式求值{sqstack1OPND;//运算数栈sqstack2OPTR;//运算符栈intn;charc,c2;Lnode*first;first=p;p=p-next;InitStack1(OPND);InitStack2(OPTR);while(!stackempy2(OPTR)||p){while(p){switch(p-data){case1:n=p-data1;Push1(OPND,n);break;case2:c=p-data2;if(stackempy2(OPTR))Push2(OPTR,c);else{switch(c){case'(':Push2(OPTR,c);break;case')':GetTop2(OPTR,c2);while(c2!='('){Operat(OPND,OPTR);GetTop2(OPTR,c2);}Popopnd2(OPTR,c2);break;default:GetTop2(OPTR,c2);if(compare(c2,c))Push2(OPTR,c);else{Operat(OPND,OPTR);Push2(OPTR,c);}break;}}break;default:printf(结点有误);break;}p=p-next;}while(!stackempy2(OPTR))Operat(OPND,OPTR);}Popopnd1(OPND,n);p=first-next;while(p){if(p-data==1)printf(%d,p-data1);if(p-data==2)printf(%c,p-data2);p=p-next;}printf(=%d,n);}voidhouzhui(Lnode*p)//中缀表达式转化为后缀表达式{sqstack2OPTR;//运算符栈Lnode*r,*q,*head;intn;charc,c2;InitStack2(OPTR);p=p-next;q=(Lnode*)malloc(sizeof(structNode));head=q;while(p){switch(p-data){case1:n=p-data1;r=(Lnode*)malloc(sizeof(structNode));q-next=r;q=q-next;q-data=1;q-data1=n;break;case2:c=p-data2;if(stackempy2(OPTR))Push2(OPTR,c);else{switch(c){case'(':Push2(OPTR,c);break;case')':Popopnd2(OPTR,c2);while(c2!='('){r=(Lnode*)malloc(sizeof(structNode));q-next=r;q=q-next;q-data=2;q-data2=c2;Popopnd2(OPTR,c2);}break;default:GetTop2(OPTR,c2);while(!compare(c2,c)){Popopnd2(OPTR,c2);r=(Lnode*)malloc(sizeof(structNode));q-next=r;q=q-next;q-data=2;q-data2=c2;GetTop2(OPTR,c2);}Push2(OPTR,c);break;}}break;default:printf(结点有误);break;}p=p-next;}while(!stackempy2(OPTR)){Popopnd2(OPTR,c2);r=(Lnode*)malloc(sizeof(structNode));q-next=r;q=q-next;q-data=2;q-data2=c2;}q-next=NULL;q=head-next;while(q){if(q-data==1)printf(%d,q-data1
本文标题:C语言实现中缀、后缀、前缀表达式-相互转化并求值
链接地址:https://www.777doc.com/doc-7254092 .html