当前位置: 首页 > news >正文

计算器字符串转换问题

计算器字符串转换问题

一. 中缀表达式和后缀表达式

  1. 中缀表达式(人类可以看懂的四则运算表达式,运算符在运算元素中间)
    ( 1 + 2 ) ∗ 3 − − 6 (1+2)*3--6 1+236

  2. 计算机计算的逻辑是,一个双目运算符对应一个运算指令,参与运算的是两个数字,所以一般表示成 如下格式

    (1+2)==》 1 2 +

    3*3 ==》 3 3 *

    9- -6 ==》 9 -6 -

二. 解决方法

  1. 将中缀表达式进行数字和运算符的分离

    包含的元素有:
    数字和小数点:0~9和.
    符号位:+或-
    运算符:+ - * /
    括号:( )
    
  2. 将中缀表达式转换为后缀表达式

    难点:
    分离数字和运算符
    正负号和加减运算符的区分
    四则运算表达式括号必须匹配
    根据运算符优先级进行转换
    
  3. 通过后缀表达式计算最终结果

三. 算法实现

  1. 参考代码

    #include "QCalculatorDec.h"
    
    QCalculatorDec::QCalculatorDec()
    {
        m_exp = "";
        m_result = "";
    }
    
    QCalculatorDec::~QCalculatorDec()
    {
    
    }
    
    bool QCalculatorDec::isDigitOrDot(QChar c)
    {
        return (('0' <= c) && (c <= '9')) || (c == '.');
    }
    
    bool QCalculatorDec::isSymbol(QChar c)
    {
        return isOperator(c) || (c == '(') || (c == ')');
    }
    
    bool QCalculatorDec::isSign(QChar c)
    {
        return (c == '+') || (c == '-');
    }
    
    bool QCalculatorDec::isNumber(QString 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;
        QQueue<QString> spExp = split(exp);
        QQueue<QString> postExp;
    
        m_exp = exp;
    
        if( transform(spExp, postExp) )
        {
            m_result = calculate(postExp);
    
            ret = (m_result != "Error");
        }
        else
        {
            m_result = "Error";
        }
    
    
        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.clear();
                }
    
                if( isSign(exp[i]) && ((pre == "") || (pre == "(") || isOperator(pre)) )
                {
                    num += exp[i];
                }
                else
                {
                    ret.enqueue(exp[i]);
                }
    
                pre = exp[i];
            }
        }
    
        if( !num.isEmpty() )
        {
            ret.enqueue(num);
        }
    
        return ret;
    }
    
    bool QCalculatorDec::match(QQueue<QString>& exp)
    {
        bool ret = true;
        int len = exp.length();
        QStack<QString> stack;
    
        for(int i=0; i<len; i++)
        {
            if( isLeft(exp[i]) )
            {
                stack.push(exp[i]);
            }
            else if( isRight(exp[i]) )
            {
                if( !stack.isEmpty() && isLeft(stack.top()) )
                {
                    stack.pop();
                }
                else
                {
                    ret = false;
                    break;
                }
            }
        }
    
        return ret && stack.isEmpty();
    }
    
    bool QCalculatorDec::transform(QQueue<QString>& exp, QQueue<QString>& output)
    {
        bool ret = match(exp);
        QStack<QString> stack;
    
        output.clear();
    
        while( ret && !exp.isEmpty() )
        {
            QString e = exp.dequeue();
    
            if( isNumber(e) )
            {
                output.enqueue(e);
            }
            else if( isOperator(e) )
            {
                while( !stack.isEmpty() && (priority(e) <= priority(stack.top())) )
                {
                    output.enqueue(stack.pop());
                }
    
                stack.push(e);
            }
            else if( isLeft(e) )
            {
                stack.push(e);
            }
            else if( isRight(e) )
            {
                while( !stack.isEmpty() && !isLeft(stack.top()) )
                {
                    output.enqueue(stack.pop());
                }
    
                if( !stack.isEmpty() )
                {
                    stack.pop();
                }
            }
            else
            {
                ret = false;
            }
        }
    
        while( !stack.isEmpty() )
        {
            output.enqueue(stack.pop());
        }
    
        if( !ret )
        {
            output.clear();
        }
    
        return ret;
    }
    
    QString QCalculatorDec::calculate(QString l, QString op, QString r)
    {
        QString ret = "Error";
    
        if( isNumber(l) && isNumber(r) )
        {
            double lp = l.toDouble();
            double rp = r.toDouble();
    
            if( op == "+" )
            {
                ret.sprintf("%f", lp + rp);
            }
            else if( op == "-" )
            {
                ret.sprintf("%f", lp - rp);
            }
            else if( op == "*" )
            {
                ret.sprintf("%f", lp * rp);
            }
            else if( op == "/" )
            {
                const double P = 0.000000000000001;
    
                if( (-P < rp) && (rp < P) )
                {
                    ret = "Error";
                }
                else
                {
                    ret.sprintf("%f", lp / rp);
                }
    
            }
            else
            {
                ret = "Error";
            }
        }
    
        return ret;
    }
    
    QString QCalculatorDec::calculate(QQueue<QString>& exp)
    {
        QString ret = "Error";
        QStack<QString> stack;
    
        while( !exp.isEmpty() )
        {
            QString e = exp.dequeue();
    
            if( isNumber(e) )
            {
                stack.push(e);
            }
            else if( isOperator(e) )
            {
                QString rp = !stack.isEmpty() ? stack.pop() : "";  //从栈中弹出右操作数
                QString lp = !stack.isEmpty() ? stack.pop() : "";  //从栈中弹出左操作数
                QString result = calculate(lp, e, rp);   //根据符号进行运算
    
                if( result != "Error" )
                {
                    stack.push(result);   //将计算结果压入栈中
                }
                else
                {
                    break;
                }
            }
            else
            {
                break;
            }
        }
    
    	//后缀表达式为空,栈中最后一个元素为一个数
        if( exp.isEmpty() && (stack.size() == 1) && isNumber(stack.top()) )
        {
            ret = stack.pop();  //计算结果 弹栈
        }
    
        return ret;
    }
    
    

相关文章:

  • QDialog - 对话框
  • 登录对话框的设计和实现
  • Qt中的标准对话框
  • 布局管理器
  • Qt工程添加资源文件(例如:图标)
  • 主窗口 (QMainWindow)
  • C++中的explicit
  • Qt中的事件处理
  • 使用二阶构造实现计算器页面的构造过程
  • Linux内核的CodingStyle
  • 常见面试题及面试准备阶段要思考的问题
  • Qt中事件的传递过程
  • Qt事件初探-发现事件
  • Qt中TCP通信的实现
  • ubuntu下MQTT的移植安装
  • 【译】理解JavaScript:new 关键字
  • 4月23日世界读书日 网络营销论坛推荐《正在爆发的营销革命》
  • Angularjs之国际化
  • Computed property XXX was assigned to but it has no setter
  • ES6语法详解(一)
  • JDK9: 集成 Jshell 和 Maven 项目.
  • JS 面试题总结
  • MobX
  • nfs客户端进程变D,延伸linux的lock
  • Octave 入门
  • Puppeteer:浏览器控制器
  • 从零开始的webpack生活-0x009:FilesLoader装载文件
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 如何使用 OAuth 2.0 将 LinkedIn 集成入 iOS 应用
  • 怎么把视频里的音乐提取出来
  • postgresql行列转换函数
  • RDS-Mysql 物理备份恢复到本地数据库上
  • #每天一道面试题# 什么是MySQL的回表查询
  • ()、[]、{}、(())、[[]]等各种括号的使用
  • (1)Android开发优化---------UI优化
  • (done) 两个矩阵 “相似” 是什么意思?
  • (poj1.2.1)1970(筛选法模拟)
  • (附源码)ssm户外用品商城 毕业设计 112346
  • (论文阅读40-45)图像描述1
  • (转)Spring4.2.5+Hibernate4.3.11+Struts1.3.8集成方案一
  • (转载)(官方)UE4--图像编程----着色器开发
  • .NET CORE 第一节 创建基本的 asp.net core
  • .net core IResultFilter 的 OnResultExecuted和OnResultExecuting的区别
  • .net core webapi 部署iis_一键部署VS插件:让.NET开发者更幸福
  • .net core控制台应用程序初识
  • .net 提取注释生成API文档 帮助文档
  • .net 无限分类
  • .NET 中小心嵌套等待的 Task,它可能会耗尽你线程池的现有资源,出现类似死锁的情况
  • .NET/C# 在代码中测量代码执行耗时的建议(比较系统性能计数器和系统时间)
  • .Net语言中的StringBuilder:入门到精通
  • .net最好用的JSON类Newtonsoft.Json获取多级数据SelectToken
  • /etc/sudoer文件配置简析
  • [ vulhub漏洞复现篇 ] JBOSS AS 4.x以下反序列化远程代码执行漏洞CVE-2017-7504
  • [ 网络基础篇 ] MAP 迈普交换机常用命令详解
  • [Bugku]密码???[writeup]