107.堆栈四则运算

举报
C语言与CPP编程 发表于 2022/05/05 00:50:13 2022/05/05
【摘要】 /* 在BC31下编译 或VC6.0*//* compile under Borland C++ 3.1 or Visual C++ 6.0*/ /*#include "stdafx.h"*/#include "stdio.h"#include "string.h"#include "stdlib.h"#include "conio.h...

  
  1. /* 在BC31下编译 或VC6.0*/
  2. /* compile under Borland C++ 3.1 or Visual C++ 6.0*/
  3. /*#include "stdafx.h"*/
  4. #include "stdio.h"
  5. #include "string.h"
  6. #include "stdlib.h"
  7. #include "conio.h"
  8. #define TRUE 1
  9. #define FALSE 0
  10. #define STACK_INIT_SIZE 100/*存储空间初始分配量*/
  11. #define STACKINCREMENT 20/*存储空间分配增量*/
  12. typedef struct
  13. {
  14. int *pBase;/*在构造之前和销毁之后,base的值为NULL*/
  15. int *pTop;/*栈顶指针*/
  16. int StackSize;/*当前已分配的存储空间,以元素为单位*/
  17. }Stack;
  18. typedef int BOOLEAN;
  19. char Operator[8]="+-*/()#";/*合法的操作符存储在字符串中*/
  20. char Optr;/*操作符*/
  21. int Opnd=-1;/*操作符*/
  22. int Result;/*操作结果*/
  23. /*算符间的优先关系*/
  24. char PriorityTable[7][7]=
  25. {
  26. {'>','>','<','<','<','>','>'},
  27. {'>','>','<','<','<','>','>'},
  28. {'>','>','>','>','<','>','>'},
  29. {'>','>','>','>','<','>','>'},
  30. {'<','<','<','<','<','=','o'},
  31. {'>','>','>','>','o','>','>'},
  32. {'<','<','<','<','<','o','='},
  33. };
  34. //数据对象的操作方法
  35. //构造一个空栈,如果返回值为0,则表示初始化失败
  36. Stack InitStack()/*这是个效率低的方法*/
  37. {
  38. Stack S;
  39. S.pBase=(int*)malloc(STACK_INIT_SIZE*sizeof(int));
  40. if(!S.pBase)
  41. {/*内存分配失败*/
  42. printf("内存分配失败,程序中止运行\n");
  43. exit(-1);
  44. }
  45. else
  46. {
  47. S.pTop=S.pBase;
  48. S.StackSize=STACK_INIT_SIZE;
  49. }
  50. return S;
  51. }
  52. //销毁栈S,S不再存在
  53. void DestoryStack(Stack *S)
  54. {
  55. if(S->pBase)
  56. {
  57. free(S->pBase);
  58. S->pTop=S->pBase=NULL;
  59. }
  60. }
  61. //若栈不空,则用e返回S的栈顶元素
  62. //注:由于应用的特殊,可以不检查栈是否为空
  63. int GetTop(Stack S)
  64. {
  65. return *(S.pTop-1);
  66. }
  67. //插入元素e为新的栈顶元素,如果成功则返回1,否则返回0
  68. int Push(Stack *S,int e)
  69. {
  70. if(S->pTop-S->pBase==S->StackSize)
  71. {//栈满,追加存储空间
  72. S->pBase=(int*)realloc(S->pBase,S->StackSize+STACKINCREMENT*sizeof(int));
  73. if(!S->pBase)
  74. return 0;//存储分配失败
  75. S->pTop=S->pBase+S->StackSize;
  76. S->StackSize+=STACKINCREMENT;
  77. }
  78. *(S->pTop++)=e;
  79. return 1;
  80. }
  81. int Pop(Stack *S,int *e)
  82. {//若栈不空,则删除S的栈顶元素,用e 返回其值,并返回1;否则返回0
  83. if(S->pTop==S->pBase)
  84. return 0;
  85. *e=*--(S->pTop);
  86. return 1;
  87. }
  88. //主函数及其它函数的实现
  89. //比较两个数学符号operator_1,operator_2的计算优先权,在算符优先关系表中查找相应的关系并返回'<','=',或'>'
  90. char CheckPriority(char operator_1,char operator_2)
  91. {
  92. int i,j;//用来查询算符间优先关系表的下标
  93. //char *ptr;
  94. i=strchr(Operator,operator_1)-Operator;//找到传入操作符在字符串Operators中的相对位置
  95. j=strchr(Operator,operator_2)-Operator;
  96. //返回算符优先关系表中相应值
  97. return PriorityTable[i][j];
  98. }
  99. BOOLEAN IsOperator(char ch)
  100. {//判断一个字符是否为打操作符
  101. if(strchr(Operator,ch))
  102. return TRUE;
  103. else
  104. return FALSE;
  105. }
  106. //从键盘获得输入
  107. void GetInput(void)
  108. {
  109. char Buffer[20];//键盘输入缓冲区,用来处理输入多位数的情况
  110. char ch;//存放键盘输入
  111. int index;//存放Buffer的下标
  112. index=0;
  113. ch=getch();//从键盘读入一个字符
  114. while(ch!=13&&!IsOperator(ch))
  115. {//如果输入的字符是回车符或是操作符,循环结束
  116. if(ch>='0'&&ch<='9')
  117. {//将字符回显到屏幕
  118. printf("%c",ch);
  119. Buffer[index]=ch;
  120. index++;
  121. }
  122. ch=getch();
  123. }
  124. if(ch==13)
  125. Optr='#';//输入的表达式以回车符结束
  126. else
  127. {
  128. Optr=ch;
  129. printf("%c",ch);
  130. }
  131. if(index>0)
  132. {
  133. Buffer[index]='\0';
  134. Opnd=atoi((Buffer));
  135. }
  136. else
  137. Opnd=-1;//程序不支持输入负数,当Opnd为负数时,表示输入的字符为操作符
  138. }
  139. //计算形如a+b之类的表达式,theta为操作符,a,b为操作数
  140. int Calc(int a,char theta,int b)
  141. {
  142. switch(theta)
  143. {
  144. case '+':
  145. return a+b;
  146. case '-':
  147. return a-b;
  148. case '*':
  149. return a*b;
  150. default:
  151. if(b==0)//除数为零的情况
  152. {
  153. printf("除数不能为");
  154. return 0;//返回0用以显示
  155. }
  156. else
  157. return a/b;
  158. }
  159. }
  160. /*表达式求值*/
  161. BOOLEAN EvaluateExpression()
  162. {
  163. int temp;//临时变量
  164. char theta;//存放操作符的变量
  165. int itheta;//存放出栈的操作符的变量add by me
  166. int a,b;//存放表达式运算时的中间值
  167. int topOpnd;//栈顶操作数
  168. char topOptr;//栈顶操作符
  169. Stack OPTR=InitStack();//操作符栈
  170. Stack OPND=InitStack();//操作数栈
  171. if(!Push(&OPTR,'#'))//操作符栈中的第一个为#字符
  172. return FALSE;
  173. GetInput();//从键盘获得输入
  174. while(Optr!='#'||GetTop(OPTR)!='#')
  175. {//如果Optr>=0,表示有操作数输入
  176. if(Opnd>=0)Push(&OPND,Opnd);
  177. switch(CheckPriority(GetTop(OPTR),Optr))
  178. {
  179. case '<'://栈顶元素优先权低
  180. if(!Push(&OPTR,Optr))return FALSE;
  181. GetInput();
  182. break;
  183. case '='://脱括号并接收键盘输入
  184. Pop(&OPTR,&temp);GetInput();
  185. break;
  186. case '>'://退栈并将运算结果入栈
  187. //先用itheta得到操作符在赋给theta
  188. Pop(&OPTR,&itheta);
  189. Pop(&OPND,&b);
  190. Pop(&OPND,&a);
  191. theta = (char)( itheta );
  192. Push(&OPND,Calc(a,itheta,b));
  193. Opnd=-1;
  194. break;
  195. }
  196. }
  197. //本算法中,当输入只有一个操作数然后就输入回车符时,
  198. //OPND.pTop==OPND.pBase
  199. //如果OPND.pTop==OPND.pBase并且Opnd<0,则说明用户
  200. //未输入任何操作和操作符而直接输入[回车],程序直接
  201. //退出运行
  202. if(OPND.pTop==OPND.pBase&&Opnd<0)
  203. {
  204. printf("\n\n感谢使用!\n");
  205. exit(1);
  206. }
  207. else if(OPND.pTop==OPND.pBase)
  208. Result=Opnd;
  209. else
  210. {
  211. Result=GetTop(OPND);
  212. DestoryStack(&OPND);
  213. DestoryStack(&OPTR);
  214. }
  215. return TRUE;
  216. }
  217. void Message(void)
  218. {
  219. printf("\n四则运算表达式求值演示\n");
  220. printf("-------------------------------\n");
  221. printf("使用方法:请从键盘上直接输入表达式,以回车键结束.如45*(12-2)[回车]\n");
  222. printf("注0:不输入任何数而直接按[回车]键,将退出程序.\n");
  223. printf("注1:本程序暂时不接受除数字键及四则运算符之外的任何其它键盘输入.\n");
  224. printf("注2:本程序暂时只能处理正确的表达式,不支持输入负数.\n");
  225. printf("-------------------------------\n\n");
  226. }
  227. void main(void)
  228. {
  229. int i;//用来一些说明性信息
  230. Message();
  231. for(i=1;;i++)
  232. {
  233. printf("表达式%d:",i);
  234. if(EvaluateExpression())
  235. printf("=%d\n",Result);
  236. else
  237. printf("计算中遇到错误\n");
  238. }
  239. }

文章来源: blog.csdn.net,作者:程序员编程指南,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/weixin_41055260/article/details/124558688

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。