计算机如何读懂四则运算表达式?
9.3 + (3 - -0.11) * 5
后缀表达式
人类习惯的数学表达式叫做中缀表达式
另外,还有一种将运算符放在数字后面的后缀表达式
5 + 3——> 5 3 +
1 + 2 * 3 ——> 1 2 3 * +
9 + (3 - 1) *5 ——> 9 3 1 - 5* +
中缀表达式符合人类的阅读和思维习惯
后缀表达式符合计算机的运算方式
——消除了中缀表达式中的括号
——同时保留中缀表达式中的运算优先级
解决方案
1.将中缀表达式进行数字和运算符的分离
2.将中缀表达式转换为后缀表达式
3.通过后缀表达式计算最终结果
分离算法分析
所要计算的中缀表达式中包含
——数字和小数点[ 0-9或.]
——符号位[+ 或-]
——运算符[+ - * /]
——括号 [( 或 )]
9.3 + ( 3 - -0.11 ) * 5
思想:以符号作为标志对表达式中的字符逐个访问
——定义累计变量num(字符串)
——当前字符exp[i]为数字或小数点时:
累计:num += exp[i]
——当前字符exp[i]为符号时:
num为运算数,分离并保存
若exp[i]为正负号:
累计符号位 +和- : num +=exp[i]
若exp[i]为运算符:
分离并保存
for(int i=0; i<exp.length(); i++) { if( exp[i]为数字或小数点) 累计:num += exp[i]; else if(exp[i]为符号) { if( num != "") 分离并保存运算数: num if(exp[i]为正号或负号) 符号位累计: num += exp[i] else { 分离并保存运算符: exp[i]; } } }
难点:
——如何区分正负号与加号和减号
+ 和 - 在表达式的第一个位置 (前一个字符为空,必然是正负号)
括号后的 + 和 - (前一个字符是括号,必然是正负号)
运算符后的 + 和 - (前一个字符是运算符,必然是正负号)
QCalculatorDec.h
#ifndef _QCALCULATORDEC_H_ #define _QCALCULATORDEC_H_ #include <QString> #include <QQueue> #include <QStack> class QCalculatorDec { protected: QString m_exp; // 代表用户输入的四则运算表达式 QString m_result; //计算结果 bool isDigitOrDot(QChar c); bool isSymbol(QChar c); bool isSign(QChar c); bool isNumber(QString s); bool isOperator(QString s); bool isLeft(QString s); bool isRight(QString s); int priority(QString s); QQueue<QString> split(const QString& exp); public: QCalculatorDec(); ~QCalculatorDec(); bool expression(const QString& exp); QString expression(); QString result(); }; #endif // _QCALCULATORDEC_H_
QCalculatorDec.cpp
#include "QCalculatorDec.h" #include <QDebug> QCalculatorDec::QCalculatorDec() { m_exp = " "; m_result = " "; //为了测试使用 QQueue<QString> r = split("-9.11 + (3 - -1)* -5"); for(int i=0; i<r.length(); i++) { qDebug() << r[i]; } } QCalculatorDec::~QCalculatorDec() { } bool QCalculatorDec::isDigitOrDot(QChar c) { return ((('0' <= c) && (c <= '9')) || (c == '.')); } bool QCalculatorDec::isSymbol(QChar c) //判读当前的字符C究竟是不是操作符或者括号 { return isOperator(c) || (c == '(') || (c == ')'); } bool QCalculatorDec::isSign(QChar c) //判断当前的字符是不是正负号 { return (c == '+') || (c == '-'); } bool QCalculatorDec::isNumber(QString s) //判断当前的s是不是合法的数字 { bool ret = false; s.toDouble(&ret); return ret; } bool QCalculatorDec::isOperator(QString s) { return (s == "+") || (s == "-") || (s == "*") || (s == "/") ; } bool QCalculatorDec::isLeft(QString s) { return (s == "("); } bool QCalculatorDec::isRight(QString s) { return (s == ")"); } int QCalculatorDec::priority(QString s) { int ret = 0; if((s == "+") || (s == "-")) { ret = 1; } if((s == "*") || (s == "/")) { ret = 2; } return ret; } bool QCalculatorDec::expression(const QString &exp) { bool ret = false; return ret; } QString QCalculatorDec::result() { return m_result; } QQueue<QString> QCalculatorDec::split(const QString &exp) { QQueue<QString> ret; QString num = ""; QString pre = ""; //用来保存前一个字符的 for(int i=0; i<exp.length(); i++) { if(isDigitOrDot(exp[i])) { num += exp[i]; pre = exp[i]; } else if(isSymbol(exp[i])) { if(!num.isEmpty()) { ret.enqueue(num); //如果不为空,就应该分离并保存了。保存到队列中,之后num就应该清空,以便累计下一个运算数。 num.clear(); } if(isSign(exp[i]) && ((pre == "") || (pre == "(") || isOperator(pre))) { num += exp[i]; } else { ret.enqueue(exp[i]); } pre = exp[i];//将这个字符保存下来,当进行下一次循环时,它将作为前一个字符使用 } } if(!num.isEmpty()) //如果for循环运行结束之后,num变量里面还有没有东西呢?如果不为空,里面还保存着最后的一个运算数。应将其分离保存到返回队列中去。 { ret.enqueue(num); qDebug() << num; } //这个地方很关键,四则运算中的最后一个操作数。 return ret; }
main.cpp
#include <QApplication> #include "QCalculatorUI.h" #include "QCalculatorDec.h" int main(int argc, char *argv[]) { #if 0 QApplication a(argc, argv); QCalculatorUI* cal = QCalculatorUI::NewInstance(); int ret = 0; if(cal != NULL) { cal->show(); ret = a.exec(); delete cal; //当程序运行到最后时,将生成的cal对象释放掉。 } return ret; #endif QCalculatorDec c; return 0; }
来源:https://www.cnblogs.com/-glb/p/12094213.html