简易计算器c++

简易计算器

  • 问题描述:用c++实现一个能够进行加减乘除运算的简易计算器。输入包括int型、'+'、'-'、'*'、'/'、'('、')',没有空格并以'='结尾的中序表达式。输出转化后的后续表达式和结果。

中序、后序表达式

  • 中序表达式符合人的思维习惯从左至右接收运算数和运算符按顺序计算,但对于计算机来说较为复杂,一旦出现了括号将会改变算符实际计算过程中的优先级

例如:3*(2+2)+1

  • 后序表达式因其没有括号、只需要单向扫描而更符合计算机的计算过程,将需要运算的数字依次入栈直到扫描到算符,则将存放运算数的栈弹出操作数进行操作,并将结果重新入栈作为新的操作数,直到表达式扫描结束,运算数栈中的元素就是最后的结果

将上述例子转化为后序表达式:3 2 2 + * 1 +

中序表达式转化为后序表达式

  • 思路:因为计算机对于后序表达式是从左向右进行扫描,所以运算符实际运算中(考虑中序表达式中有括号的情况)的优先级决定了运算符在后序表达式中的位置。对于操作数没有所谓的优先级,所以遇到操作数直接输出为后序表达式元素即可。

  • 运算符入栈:由于后序表达式中的运算符位于操作数的后边,所以在转化过程中应当先把运算符存放在一个栈中。
    那么,弹出条件:

    1. 操作数全部输出,中序表达式扫描结束,运算符依次弹出;
case '=':
                while (!temp.empty())
                {
                    calBefore.push_back(temp.top());
                    calBefore.push_back(' ');
                    temp.pop();
                }
                break;
  1. 新入栈元素优先级低于栈顶元素时,需要先将栈顶元素弹出为后序表达式元素,再将新运算符入栈,以保证后序表达式中的运算符顺序与计算优先级一致。
if (!temp.empty())
{
  while (!temp.empty() && (temp.top() == '/' || temp.top() == '*'))
  {
   calBefore.push_back(temp.top());
  calBefore.push_back(' ');
  temp.pop();
  }
}
temp.push(*it);
break;
  • 括号处理:中序表达式中用括号改变了运算符的优先级,转化过程中,需要将括号当作运算符入栈以此区分括号内外运算符。
    1.‘(’无条件入栈;
    1. 扫描至‘)’时,需要不断将栈中元素弹出直至将‘(’弹出。注意:后序表达式中没有括号,括号不需要输出
  • 举例说明:



    根据上述例子 3*(2+2)+1 可以画出上图,左边是string类型的中序表达式,中间是stack类型存放运算符,右边是string类型用于输出的后序表达式。

  1. 按步骤操作至上图都不存在需要特殊处理的操作:操作数直接输出为后序表达式、运算符入栈,每次入栈的运算符都比栈顶元素优先级低(或相等)
    2.中序表达式下一个元素为 ‘)’,应当将运算符栈在 ‘ (’之后入栈的元素全部弹出,此处只有一个元素为 “+”,则得到下图



    3.接着下一个元素为 “+”,入栈发现优先级低于栈顶元素 “*”,则要将 “*”先弹出再入栈,得下图

    4.下一个元素为“1”,直接输出。最后若运算符栈不为空全部弹出。
    得到后序表达式为 3 2 2 + * 1 +。

由后序表达式计算结果

将中序表达式转化为后序表达式后,对于计算机来说计算就变得很简单了:
1.只需要从左往右扫描表达式,将操作数存入栈中等待运算
2.扫描到运算符就将操作数栈顶部两个操作数弹出参与运算,运算结果重新入栈作为新的操作数
3.如此执行,直到表达式扫描完成,操作数中的最后一个数就是计算结果


如图所示,将操作数入栈直至碰到运算符 ‘+’,此时将操作数栈顶部两个元素弹出,分别是 ‘2’、‘2’,运算结果 ‘4’ 重新入栈得下图


此时扫描至表达式中下一位 ‘ * ’,同样弹出操作数栈顶部两个元素分别是 ‘ 4 ’、‘ 3 ’,参与运算得到 ‘ 12 ’入栈,得下图



继续 ——‘ 1 ’入栈,得下图



扫描至 ‘+’ 号弹出操作数运算,得 ‘ 13 ’入栈,得下图

表达式扫描完毕,操作数栈中元素即为运算结果。

代码如下

int Calculation(string calstr)
    {
        stack<int> calnum;
        string calBefore_ = calstr;

        string::iterator it = calBefore_.begin();
        while (!calBefore_.empty())
        {
            if (*it == 32)  //读取空格
            {
                it = calBefore_.erase(calBefore_.begin());
                continue;
            }
            else if (48 <= *it && *it <= 57)
            {
                string temp;
                int i = 0;
                while (48 <= *it && *it <= 57)
                {
                    temp.push_back(*it);
                    it++;
                    i++;
                }
                while (i > 0)
                {
                    it = calBefore_.erase(calBefore_.begin());
                    i--;
                }
                calnum.push(atoi(temp.c_str()));
                continue;
            }
            else
            {
                switch (*it)
                {
                case '+':
                    calnum.push(Add(calnum));
                    break;
                case '-':
                    calnum.push(Minus(calnum));
                    break;
                case '*':
                    calnum.push(Mutiply(calnum));
                    break;
                case '/':
                    calnum.push(Division(calnum));
                    break;
                default:
                    cout << "Error!" << endl;
                    break;
                }
                it = calBefore_.erase(calBefore_.begin());
            }
        }

        return calnum.top();
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,509评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,806评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,875评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,441评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,488评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,365评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,190评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,062评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,500评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,706评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,834评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,559评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,167评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,779评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,912评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,958评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,779评论 2 354

推荐阅读更多精彩内容

  • Java用栈实现综合计算器 栈 栈(stack)又名堆栈,它是一种运算受限的线性表 。限定仅在表尾进行插入和删除...
    coderZ666阅读 449评论 0 0
  • 一、定义 栈是一种线性表结构,栈结构中有两端,对栈的操作都是对栈的一端进行操作的,那么被操作的一端称为栈顶,另一端...
    rainple阅读 703评论 1 1
  • 程序设计基础(上) 前言 C++的常见错误 语法错误: 混淆结构体类型与结构体变量的区别,为一个结构体类型赋值。不...
    修Nya阅读 1,264评论 0 0
  • 点赞关注,不再迷路,你的支持对我意义重大!Hi,我是丑丑。本文「数据结构 & 算法」| 导读 —— 登高博见[ht...
    彭旭锐阅读 443评论 0 1
  • 栈是一个先进后出的一种数据结构对于给定一个字符串计算其值,其字符串可能有三种:前,中和后缀表达式而对于这三种表达式...
    墨宇暗黑阅读 190评论 0 0