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的更多相关文章

  1. 《Java程序设计》十四次作业

    <Java程序设计>十四次作业实验总结 1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结与数据库相关内容. 3. 代码量统计 周次 总代码量 新增代码量 总文件数 新增 ...

  2. 2016-2017-2 20155339《 java面向对象程序设计》实验四Android程序设计

    2016-2017-2 20155339< java面向对象程序设计>实验四Android程序设计 实验内容 1.Android Stuidio的安装测试: 参考<Java和Andr ...

  3. sdut 面向对象程序设计上机练习四(变量引用)

    面向对象程序设计上机练习四(变量引用) Time Limit: 1000MS Memory limit: 65536K 题目描写叙述 将变量的引用作为函数形參,实现2个int型数据交换. 输入 输入2 ...

  4. C语言程序设计实验第四次作业

    (一)改错题 输出三角形的面积和周长,输入三角形的三条边a.b.c,如果能构成一个三角形,输出面积area和周长perimeter(保留2位小数):否则,输出"These sides do ...

  5. 《面向对象程序设计》第三次作业 Calculator

    c++第三次作业 Calculator git上的作业展示点这里. ps:有一点不是很明确,作业要求:将数字和符号提取出来,得到一组string,然后才将这些string存入队列中.按我的理解是需要将 ...

  6. 面向对象课程 - 寒假第四次作业 - C++计算器项目计算部分

    C++计算器项目计算部分 零.项目源文件地址 地址:Calculator 2.0 一.项目信息相关 项 目 : Calculator 版 本 : 2 . 0 日 期 : 2016 . 4 . 14 实 ...

  7. 面向对象程序设计(C++)_作业一_设计、定义并实现Complex类

    源代码: 运行结果:

  8. THE LAST ONE!! 2017《面向对象程序设计》课程作业八

    THE LAST ONE!! 2017<面向对象程序设计>课程作业八 031602230 卢恺翔 GitHub传送门 题目描述 1.时间匆匆,本学期的博客作业就要结束了,是否有点不舍,是否 ...

  9. 面向对象程序设计第四单元总结(UML系列)

    2019面向对象程序设计第四单元总结 前言 ​ 本单元是面向对象程序设计课程的最后一个单元了,本单元是和UML模型相关,也就是说,我们需要正确理解UML模型的基础上,对构建出的UML模型进行解析,但是 ...

随机推荐

  1. [转].Net实现本地化简易教程

    本文转自:https://www.cnblogs.com/csdbfans/archive/2011/10/17/2214048.html 实现多语言版本的支持,就是所谓的国际化,也说是本地化. 今天 ...

  2. GetHashCode方法学习

    GetHashCode方法我的理解是做两个对象的比较,每个对象,不管是值类型还是应用类型都提供这个基本函数,都可以去重写它.GetHashTable通常用于HashTable.List<> ...

  3. 解决XShell不能使用小键盘的问题

    新建链接的时候,在Terminal节点,选择VT Modes为set to normal.

  4. JPA注解@GeneratedValue

    @GeneratedValue是JPA的标准用法, JPA提供四种标准用法,由@GeneratedValue的源代码可以看出. public enum GenerationType { TABLE, ...

  5. 地址解析协议ARP,网络层协议IP、ICMP协议

    分析所用软件下载:Wireshark-win32-1.10.2.exe 阅读导览 1. 分析并且应用ARP协议 2.分析IP协议 3.分析ICMP协议 1.分析arp报文的格式与内容 (1)ping ...

  6. K8S基础概念

    一.核心概念 1.Node Node作为集群中的工作节点,运行真正的应用程序,在Node上Kubernetes管理的最小运行单元是Pod.Node上运行着Kubernetes的Kubelet.kube ...

  7. vs中发布WebSever时启用HTTP-POST和HTTP-GET这两种协议

    一.问题介绍 在vs中建立一个websever项目时候默认是禁用HTTP-POST和HTTP-GET这两种协议的.但是如果你是在本机上去调试或者是在iis中浏览都会有HTTP-POST这种方式,因为这 ...

  8. Android上实现视频录制

    首先,我们肯定要用到摄像头,因此需要在Manifest文件中声明使用权限: <uses-permission android:name="android.permission.CAME ...

  9. 填报表导出excel后不可写的单元格处于锁定状态

     填报表单元格分为可写和不可写两种状态,当填报表在web上展现的时候可写单元格可以进行数据填报和修改,非可写单元格不可操作. 报表导出为excel时,润乾导出excel包默认情况下不对excel单 ...

  10. 分布式文件系统比较出名的有HDFS  和 GFS

    分布式文件系统比较出名的有HDFS  和 GFS,其中HDFS比较简单一点.本文是一篇描述非常简洁易懂的漫画形式讲解HDFS的原理.比一般PPT要通俗易懂很多.不难得的学习资料. 1.三个部分: 客户 ...