《面向对象程序设计》c++第四次作业___calculator plus
c++第四次作业 Calculator Plus
git上的作业展示 Calculator 2.0 SourceCode in Git
PS:这次作业orz感谢某同学用windows的dev c++帮我把代码编译成可执行文件,同时感谢某学长帮我克服了sourcetree上的疑难问题。(连在命令行上的截图都是帮我编译的小伙伴帮忙的)
我的计算器的一些特点(以下特点将在下文有操作范例):
1、数字不合法(整数位大于10位)报错(
ERROR:Number Not Conform To The Requirement.)2、除以零报错(
ERROR:Divided By Zero.)3、9(2+3)形式的表达在括号前自动填充乘号
4、2*-3形式的表达负数可不加括号(与普通科学计算器相同)
5、左右括号数不一样多,自动在相应端补齐括号(如:-(3-(-(2+3)+4 等同于-(3-(-(2+3)+4)) )
6、引入次方计算(如:2^3 等于 2的3次方)
我的计算器的一些不足:
1、没有使用大多数同学的前缀表达式、后缀表达式
2、没能实现四则运算与乘方以外的运算
这里有些特别想说的是,我的代码最希望与别人分享的内容并不是那些实现了计算器功能的部分,而是那些注释掉的过程输出,那些语句一方面体现了程序编码过程中艰辛的调试过程,而且一个编码人对自己的代码进行检测、修改过程中的输出更能体现编程的思想、对自己代码的掌控,所以保留了那部分点代码,仅仅注释掉。也希望更多同行可以和我分享更好的调试方法。
代码:
Scan.hpp
#ifndef Scan_hpp
#define Scan_hpp
#include <iostream>
#include <queue>
#include <string>
#include <cmath>
using namespace std;
class Scan {
private:
string input;
string charString;
public:
bool tooBig;
queue<string> ToStringQueue(string input,int mol);
};
#endif
Scan.cpp
这个文件生成两个队列,分别用于输出表达式和送入 Calculation 类中计算,分别对应参数1和2
#include "Scan.hpp"
#include <iostream>
#include <queue>
#include <string>
#include <cmath>
using namespace std;
queue<string> Scan::ToStringQueue(string input,int mol) {
tooBig=false;
input[input.length()]='+';
int count=0,flag=0,bracket=0;
string item = "";
std::queue<string> outCharQueue,calculationCharQueue;
for (int i=0; i<input.length(); i++) {
if (input[i]=='(') {
bracket++;
}else if (input[i]==')') {
bracket--;
}
}
if (bracket<0) {
for (int i=0; i>bracket; i--) {
calculationCharQueue.push("(");
}
}
for (int i=0; i<input.length(); i++) {
if ((i==0 && input[i]=='-')||(input[i-1]=='(' && input[i]=='-')) {
outCharQueue.push("-");
calculationCharQueue.push("0");
calculationCharQueue.push("-");
calculationCharQueue.push("1");
calculationCharQueue.push("*");
}
else if ((input[i]=='-')&&(input[i-1]=='+' ||input[i-1]=='-' ||
input[i-1]=='*' ||input[i-1]=='/' || input[i-1]=='^')) {
outCharQueue.push("-");
calculationCharQueue.push("(");
calculationCharQueue.push("0");
calculationCharQueue.push("-");
calculationCharQueue.push("1");
calculationCharQueue.push("*");
flag=1;
}
else if (input[i]=='+' || input[i]=='-'|| input[i]=='*' ||
input[i]=='/' || input[i]=='^'||input[i]=='(' || input[i]==')' ) {
if(!item.empty()){
outCharQueue.push(item);
calculationCharQueue.push(item);
}
item.clear();
count = 0;
if (flag==1) {
calculationCharQueue.push(")");
flag=0;
}
if (input[i]=='(' && input[i-1]>='0' && input[i-1]>='9') {
calculationCharQueue.push("*");
}
item = input[i];
outCharQueue.push(item);
calculationCharQueue.push(item);
item.clear();
}
else if ((input[i] >= 48 && input[i] <=57) || input[i] == '.' ) {
item += input[i];
if (input[i] != '.') {
count++;
}
if (count >= 10) {
tooBig = true;
}
}
}
if (!item.empty()) {
outCharQueue.push(item);
calculationCharQueue.push(item);
if (flag==1) {
calculationCharQueue.push(")");
flag=0;
}
}
if (bracket>0) {
for (int i=0; i<bracket; i++) {
calculationCharQueue.push(")");
}
}
calculationCharQueue.push("|");
if (mol==2) return calculationCharQueue;
return outCharQueue;
}
Calculation.hpp
#ifndef Calculation_hpp
#define Calculation_hpp
#include <iostream>
#include <sstream>
#include <queue>
#include <stack>
#include <cmath>
using namespace std;
class Calculation {
private:
queue<string> q;
stack<double> num;
stack<string> op;
public:
int idbz;
int oplevel(string op);
int check(string cur);
double toDouble(string num);
double calculating(queue<string> q);
double solve(double num1,double num2,string op);
};
#endif
Calculation.cpp
这个文件中有很多保留的过程输出语句
solve()函数里的第一行语句可以查看进入函数的两个运算数与一个运算符、倒数第三行语句可以查看计算过后的结果,作用在于查看每次的计算是不是你所想预想的。
calculating()函数里的类似 cout << q.front() << " &push A num" << endl 这样的语句输出每次进入运算数栈、操作符栈的变量是不是你所预想的。
- 1、solve()函数debug语句的使用

- 2、calculating()函数debug语句的使用

#include "Calculation.hpp"
#include <iostream>
#include <sstream>
#include <queue>
#include <stack>
#include <cmath>
int Calculation::oplevel(string op) {
idbz=0;
int level;
if (op[0]=='+' || op[0]=='-') {
level=2;
}else if (op[0]=='*' || op[0]=='/') {
level=3;
}else if (op[0]=='^') {
level=4;
}else if (op[0]==')' || op[0]=='(') {
level=1;
}else if (op[0]=='|') {
level=0;
}
return level;
}
double Calculation::toDouble(string num) {
stringstream s;
double number=0;
s << num;
s >> number;
return number;
}
double Calculation::solve(double num1, double num2, string op) {
// cout << num1 << " " << num2 << " " << op << " ";
if (op[0]=='+') {
num1=num1+num2;
}
if (op[0]=='-') {
num1=num1-num2;
}
if (op[0]=='*') {
num1=num1*num2;
}
if (op[0]=='/') {
if (num2!=0) {
num1=num1/num2;
}
else {
idbz=1;
return 0;
}
}
if (op[0]=='^') {
num1=pow(num1, num2);
}
// cout << num1 << endl;
return num1;
}
double Calculation::calculating(queue<string> q) {
double result=0,num1,num2;
string cur,curop;
while (!q.empty()) {
cur=q.front();
if (cur[0]=='(') {
op.push(cur);
q.pop();
if (q.front()[0]!='(') {//123
num.push(toDouble(q.front()));//123
// cout << q.front() << " &push A num" << endl;
q.pop();
}//123
}
else {
if (num.empty()) {
num.push(toDouble(cur));
// cout << cur << " &push B num" << endl;
q.pop();
cur=q.front();
}
while (!op.empty()) {
if (oplevel(cur)>oplevel(op.top()) || op.top()[0]=='(') break;
num2=num.top();num.pop();
num1=num.top();num.pop();
curop=op.top();op.pop();
// cout << num1 << " " << num2 << " " << curop << endl;
num1=solve(num1, num2, curop);
if (num1==0 && idbz==1) {
return 0;
}
num.push(num1);
// cout << num1 << " &push D num" << endl;
}
if (cur[0]==')' && op.top()[0]=='(') {
op.pop();
q.pop();
}else if (cur[0]=='|') {
break;
}else {
op.push(cur);
// cout << cur << " &push E op" << endl;
q.pop();
if (q.front()[0]!='(') {
num.push(toDouble(q.front()));
// cout << q.front() << " &push C num" << endl;
q.pop();
}
}
}
}
result=num.top();num.pop();
return result;
}
Print.hpp
#ifndef Print_hpp
#define Print_hpp
#include <iostream>
#include <queue>
#include <string>
#include <cmath>
using namespace std;
class Print {
public:
void printString(queue<string> charQueue,int mol);
};
#endif
Print.cpp
#include "Print.hpp"
#include <iostream>
#include <queue>
#include <string>
#include <cmath>
void Print::printString(queue<string> charQueue,int mol) {
if (mol==1) {
while (!charQueue.empty()) {
cout << charQueue.front() << endl;
charQueue.pop();
}
cout << endl;
}
else if (mol==2){
while (!charQueue.empty()) {
cout << charQueue.front();
charQueue.pop();
}
cout << " = " ;
}
}
main.cpp
#include <iostream>
#include <queue>
#include <string>
#include <cmath>
#include "Print.hpp"
#include "Calculation.hpp"
#include "Scan.hpp"
using namespace std;
int main(int argc,char *argv[]) {
string input;
Scan m_scan;
Print m_pr;
Calculation m_cal;
// getline(cin,input);
if (argc>1) {
if (argc==2) {
input=*(argv+1);
m_scan.ToStringQueue(input, 2);
if (m_scan.tooBig==true) {
cout << "ERROR:Number Not Conform To The Requirement." << endl;
}
else {
m_cal.calculating(m_scan.ToStringQueue(input,2));
if (m_cal.idbz==1) {
cout << "ERROR:Divided By Zero." << endl;
}else if (m_cal.idbz==0){
cout << m_cal.calculating(m_scan.ToStringQueue(input,2)) << endl;
}
}
}else if (argc==3) {
input=*(argv+1);
if (input=="-a") {
input=*(argv+2);
m_pr.printString(m_scan.ToStringQueue(input,1),2);
}
input=*(argv+2);
m_scan.ToStringQueue(input, 2);
if (m_scan.tooBig==true) {
cout << "ERROR:Number Not Conform To The Requirement." << endl;
}
else {
m_pr.printString(m_scan.ToStringQueue(input,1),2);
m_cal.calculating(m_scan.ToStringQueue(input,2));
if (m_cal.idbz==1) {
cout << "ERROR:Divided By Zero." << endl;
}else if (m_cal.idbz==0){
cout << m_cal.calculating(m_scan.ToStringQueue(input,2)) << endl;
}
}
}
}
return 0;
}
- 1、数字不合法(整数位大于10位)报错

- 2、除以零报错

- 3、9(2+3)形式的表达在括号前自动填充乘号

- 4、2*-3形式的表达负数可不加括号(与普通科学计算器相同)

- 5、左右括号数不一样多,自动在相应端补齐括号

对比

- 6、纯计算结果

- 7、-a计算结果

《面向对象程序设计》c++第四次作业___calculator plus的更多相关文章
- 《Java程序设计》十四次作业
<Java程序设计>十四次作业实验总结 1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结与数据库相关内容. 3. 代码量统计 周次 总代码量 新增代码量 总文件数 新增 ...
- 2016-2017-2 20155339《 java面向对象程序设计》实验四Android程序设计
2016-2017-2 20155339< java面向对象程序设计>实验四Android程序设计 实验内容 1.Android Stuidio的安装测试: 参考<Java和Andr ...
- sdut 面向对象程序设计上机练习四(变量引用)
面向对象程序设计上机练习四(变量引用) Time Limit: 1000MS Memory limit: 65536K 题目描写叙述 将变量的引用作为函数形參,实现2个int型数据交换. 输入 输入2 ...
- C语言程序设计实验第四次作业
(一)改错题 输出三角形的面积和周长,输入三角形的三条边a.b.c,如果能构成一个三角形,输出面积area和周长perimeter(保留2位小数):否则,输出"These sides do ...
- 《面向对象程序设计》第三次作业 Calculator
c++第三次作业 Calculator git上的作业展示点这里. ps:有一点不是很明确,作业要求:将数字和符号提取出来,得到一组string,然后才将这些string存入队列中.按我的理解是需要将 ...
- 面向对象课程 - 寒假第四次作业 - C++计算器项目计算部分
C++计算器项目计算部分 零.项目源文件地址 地址:Calculator 2.0 一.项目信息相关 项 目 : Calculator 版 本 : 2 . 0 日 期 : 2016 . 4 . 14 实 ...
- 面向对象程序设计(C++)_作业一_设计、定义并实现Complex类
源代码: 运行结果:
- THE LAST ONE!! 2017《面向对象程序设计》课程作业八
THE LAST ONE!! 2017<面向对象程序设计>课程作业八 031602230 卢恺翔 GitHub传送门 题目描述 1.时间匆匆,本学期的博客作业就要结束了,是否有点不舍,是否 ...
- 面向对象程序设计第四单元总结(UML系列)
2019面向对象程序设计第四单元总结 前言 本单元是面向对象程序设计课程的最后一个单元了,本单元是和UML模型相关,也就是说,我们需要正确理解UML模型的基础上,对构建出的UML模型进行解析,但是 ...
随机推荐
- WCF异常信息
1.服务“CJ.Demo.Conso.WcfService.EmployeeMngService”有零个应用程序(非基础结构)终结点.这可能是因为未找到应用程序的配置文件,或者在配置文件中未找到与服务 ...
- 零基础学C#算法(零基础学算法——C#版)
今天本人正在看算法方面的书.作为高中数学忘得差不多的渣渣,实在无力.无奈找了本书,c语言写的,哎.我就把其中代码翻译成C#版好了.此坑能否填平,看我耐性和网络支持条件吧.有生之年能看完的话我会把整个项 ...
- git 查看暂存区
一.简介 git ls-files 命令是用来查看暂存区中文件信息 二.常用参数 参数说明(括号里是简写) --cached(-c)显示暂存区中的文件,git ls-files命令默认的参数 --de ...
- hdu 1043 八数码问题
Eight Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Subm ...
- Java的工厂模式(三)
除了一般的工厂模式之外,还有抽象工厂模式,抽象工厂模式更强调产品族的概念,一个具体工厂生产出来的系列商品都是一个产品族的. 假设我们有两个具体工厂,分别是袋装水果工厂和罐装水果工厂,它们都能生产苹果和 ...
- Java的Collection.sort()方法
Java中的Collection.sort()方法能使用泛型对对象的变量进行排序,下面是两种方法. 文件名:student.java import java.util.*; import com.su ...
- WinForm中使用CrystalReport水晶报表——基础,分组统计,自定义数据源
开篇 本篇文章主要是帮助刚开始接触CrystalReport报表的新手提供一个循序渐进的教程.该教程主要分为三个部分1)CrystalReport的基本使用方法:2)使用CrystalReport对数 ...
- h5笔记
标签 更语义化标签 header标签 nav标签 section标签 article标签 aside标签 widget标签 footer标签 为什么要有语义化标签 能够便于开发者阅读和写出更优雅的代码 ...
- axTOCControl右键
private void axTOCControl1_OnMouseDown(object sender, ITOCControlEvents_OnMouseDownEvent e) { if (e. ...
- 地图的可视化--Folium
1.安装folium pip install MarkupSafe-0.23-cp34-none-win_amd64.whl pip install Jinja2-2.8-py2.py3-none-a ...