Calculator(1.5)

Github链接

ps.负数的处理未完成

解题过程中遇到的困难和解决

  1. <stack>的使用:

    认真研究了栈,基本上掌握了用法,与<queue>的区别就是队列中可以直接访问首元素而栈并不能。(debug过程让我深深体会到了使用s.pop()时一定要确认是否为空栈!!!!!)
  2. <sstream>的使用

    <sstream>用起来比较简单,网上查了资料就可以用了,注意点就是多次转化时要记得stream.clear,不然容易出错。
  3. 后缀表达式的转化及运算

    这次作业主要就是让我们理解前中后缀表达式吧...找到了一篇比较好理解的教程,总的过程比较长顺利,满足以下四个原则,照猫画虎地就转化了后缀表达式。

    转化原则

    • 当读到一个操作数时,立即将它放到输出中。操作符则不立即输出,放入栈中。遇到左圆括号也推入栈中。
    • 如果遇到一个右括号,那么就将栈元素弹出,将符号写出直到遇到一个对应的左括号。但是这个左括号只被弹出,并不输出。
    • 在读到操作符时,如果此时栈顶操作符优先性大于或等于此操作符,弹出栈顶操作符直到发现优先级更低的元素位置。除了处理')'的时候,否则决不从栈中移走'('。操作符中,'+' '-'优先级最低,'(' ')'优先级最高。
    • 如果读到输入的末尾,将栈元素弹出直到该栈变成空栈,将符号写到输出中。
  4. 命令行的使用

    依旧是找!资!料!掌握了新技能,开心。

一点感想

看到题目我就懵逼了,加上拖延症晚期,4.13才开始动手...总算把大部分写完了,还有个负数的情况还没搞定,过两天再填坑吧。感觉还是挺有成就感的,希望自己以后能勤快点吧...

附上代码

Calculation.cpp

#include "Calculation.h"

Calculation::Calculation()
{ } /*将Scan中的字符串传入*/
void Calculation::GetQueue(queue<string>tmp)
{
Q = tmp;
} /*后缀转化是判断是否为括号*/
bool isBr(string c)
{
if(c == "(" || c == ")")
return 1;
else
return 0;
} /*确定运算符的优先级*/
int getPri(string c)
{
if(c == "+" || c == "-")
return 1;
else if(c == "*" || c == "/")
return 2;
else if(c == "(" || c == ")")
return 0; /*将括号的优先级设为最低防止误操作*/
} /*根据运算符进行计算*/
double Cal(char c , double a , double b)
{
switch(c)
{
case'+':return(a + b);break;
case'-':return(a - b);break;
case'*':return(a * b);break;
case'/':return(a / b);break;
}
} /*后缀转化*/
void Calculation::Change(queue<string>Q)
{
while(Q.size() != 0)
{
string c;
c = Q.front(); /*处理运算符*/
if(c == "+" || c == "-" || c == "*" || c == "/" || c == "(" || c == ")")
{ /*传入括号时*/
if(isBr(c))
{
/*左括号不做处理*/
if(c == "(")
{
sym.push(c);
}
else
{
/*发现右括号时将栈顶元素弹出知道左括号,删除左括号*/
while(sym.top() != "(")
{
get.push(sym.top());
sym.pop();
}
sym.pop();
}
}
/*传入其他运算符时*/
else
{
/*根据运算符优先级进行弹出或传入*/
while(sym.size() != 0 && getPri(c) <= getPri(sym.top()))
{
get.push(sym.top());
sym.pop();
}
sym.push(c);
}
}
/*数字直接传入*/
else
{
get.push(c);
}
Q.pop();
} /*Q为空栈时,将sym栈内所有运算符弹出*/
while(sym.size() != 0)
{
get.push(sym.top());
sym.pop();
}
} /*计算过程*/
void Calculation::Ans()
{
std::stringstream stream;
double tmp;
double a;
stack<double>ans; /*存储运算值*/ /*从队首元素开始进行计算*/
while(get.size() != 0)
{
/*遇到数字直接转化为double类型入栈*/
if(get.front()[0] >= '0' && get.front()[0] <= '9')
{
stream.clear();
stream << get.front();
stream >> tmp;
ans.push(tmp);
get.pop();
}
/*遇到运算符取出栈内两个数字进行运算后再入栈*/
else
{
tmp = ans.top();
ans.pop();
a = Cal(get.front()[0] , ans.top() , tmp); /*Cal函数进行具体计算*/
ans.pop();
ans.push(a);
get.pop();
}
}
cout << ans.top() << endl; /*输出结果*/
}

最新文章

  1. OpenJudge 2985数字组合 解析报告/DP
  2. 使用cachemanager做缓存(Session的缓存)
  3. HDU 5025
  4. Ubuntu 14.04 开启启动器图标最小化功能
  5. 性能相差极大的SQL语句
  6. 【算法Everyday】第三日 KMP算法
  7. linux上备份Oracle时EXP-00091的错误解决方法
  8. 什么是流利语法Fluent Syntax
  9. 《JS权威指南学习总结--第六章 对象》
  10. Jenkins在windows上的安装配置
  11. oracle基本查询语句总结
  12. JavaScript 数组基本操作
  13. hive入门(一)、什么是hive
  14. 新概念英语(1-105)Full Of Mistakes
  15. C# Post方式下,取得其它端传过来的数据
  16. jfinal头像裁剪上传服务器
  17. effective c++ 笔记 (1-3)
  18. AOP不起作用的原因之一
  19. Informatica bulk和normal模式
  20. C# 日期和时间的字符串表示形式转换为其等效的DateTime(stringToDateTime)

热门文章

  1. python进阶(四)---需要了解的魔法方法
  2. Latex图片显示问题(1)
  3. KEEPALIVED
  4. Javascript 知识点整理
  5. 工作中最常用的Excel函数公式大全
  6. mybatis原理
  7. 理解tcp协议的3次握手和面向连接
  8. Head First 设计模式-- 总结
  9. 冰球项目日志1-yjw
  10. Java开源BI系统介绍(转)