C++,用栈的方式实现中缀表达式(四则运算)的运算,包含小数,不包含负数。


大致思路

用栈的方式实现中缀表达式的运算,因为其中涉及到小数(未涉及到负数),所以采用常规思路之前首先要对给出的中缀表达式做出一些处理。首先把给出的中缀表达式(字符串)转换成好处理的double数组,再将double数组转换为前缀的形式,最后再利用栈来计算前缀表达式


一、将字符串转换为double数组

这里需要运用到一个函数atof,但需要注意此函数的运用方法,他会把字符串中的数字转换为double型,但仅仅转换到下一个非数字字符。这么说可能不好理解,例如,给出一个字符串"1.2+2.3",如果我们把这个字符串传入这个函数,那么我们只能得到1.2这一个数字,后面的2.3是得不到的,atof函数在遇到+号的时候就已经退出了。下面是我的实现方法:

char b;
char c[20] = { '\0' };
double num[100] = { 0 };
int k = 0, len = 0;
int flag = 0;
while (true) {     //第一步,将表达式字符串转为double型的数组
        scanf("%c", &b);
        if (b == '\n') {
            if (flag == 1) {
                num[len++] = atof(c);
            }
            break;
        }
        if (b >= '0' && b <= '9' || b == '.') {
            c[k++] = b;
            flag = 1;
        }
        else if (b < '0' && flag == 1) {
            num[len++] = atof(c);
            flag = 0;
            for (int i = 0; i < k; i++) {         //清空字符串c
                c[i] = 0;
            }
            k = 0;
        }
        if (b < '0' && flag == 0) {
            num[len++] = b - '0';
        }
    }

字符b和字符串c都是中间操作,num数组才是我们需要的。注意此时运算符由于被减去了一个‘0’,所以此时是以负数形式储存在num里面,例如我们输入1.2+2.3,那么num数组里面储存的便是1.2,-5,2.3,这三个数据。

二、将中缀表达的double数组转换为前缀表达

a数组是上面得到的num数组,n是上面的num数组长度len,栈是自己写的,不是直接调用的头文件,所以函数名和头文件里面不太一样,pop是获取栈顶元素,getTop是出栈,push是入栈。此处需要准备两个栈,一个数据栈,一个符号栈。

int len;
double num1[100] = { 0 };
int len1 = 0;

void change(double a[], int n) {     //中缀换前缀
    Stack idx(n);                     //放数据的栈
    Stack symbol(n);                  //放符号的栈
    for (int i = n - 1; i >= 0; i--) {
        if (a[i] < 0) {                 //小于0说明遇到了符号
            int flag = 0;
            char temp = a[i] + '0';
            if (temp == '+' || temp == '-') {
                flag = 1;
                while (flag == 1) {
                    if ((symbol.pop() + '0') == '*' || (symbol.pop() + '0') == '/') {  //比较符号优先级
                        idx.push(symbol.getTop());
                    }
                    else {
                        flag = 0;
                        symbol.push(a[i]);
                    }
                }
            }
            else if (temp == '(') {
                while (symbol.pop() != (')' - '0')) {        //当遇到右括号时,将symbol栈里面的元素放到idx栈
                    idx.push(symbol.getTop());               //此步骤一直持续到遇到symbol栈内的下一个左括号为止
                }                                            //或者持续到栈内空
                symbol.getTop();                             //将左括号出栈
            }
            else {
                symbol.push(a[i]);                           //入栈
            }
        }
        else {
            idx.push(a[i]);
        }
    }
    while (symbol.isEmpty() == false) {
        idx.push(symbol.getTop());
    }
    while (idx.isEmpty() == false) {
        num1[len1] = idx.getTop();
        len1++;
    }
}

三、对前缀表达式进行计算

a是上面得到的num1数组,n是上面的num1数组长度len1。

double calculate(double a[], int n) {                   //波兰表达式计算,n为数据和符号的个数
    Stack s(n);
    for (int i = n - 1; i >= 0; i--) {
        if (a[i] < 0) {
            char flag = a[i] + '0';
            double b = s.getTop();
            double c = s.getTop();
            double m = 0;
            if (flag == '+') {
                m = b + c;
            }
            else if (flag == '-') {
                m = b - c;
            }
            else if (flag == '*') {
                m = b * c;
            }

            else if (flag == '/') {
                m = b / c;
            }
            s.push(m);
        }
        else {
            s.push(a[i]);
        }
    }
    return s.getTop();
}

完整代码如下:

因为是大一学C++,所以栈是老师要求我们自己写的。

#define _CRT_SECURE_NO_WARNINGS 1
#include <cstddef>
#include <iostream>
#include<iomanip>
#include<string.h>

using namespace std;

class Stack {
private:
    int top;
    double* data;
    int msize;
public:

    Stack(int n);
    ~Stack();
    void push(double e);
    double pop();
    double getTop();
    bool isEmpty();
};

Stack::Stack(int n) {               //构建实例
    msize = n;
    data = new double[msize];
    top = 0;
}

Stack::~Stack() {                   //消除实例
    delete[]data;
}

void Stack::push(double e) {       //push入栈
    if (top >= msize) {
        return;
    }
    data[top++] = e;
}

double Stack::pop() {              //pop获取栈顶元素
    if (isEmpty() == false) {
        return data[top - 1];
    }
}

double Stack::getTop() {          //getTop出栈
    if (isEmpty() == false) {
        return data[--top];
    }
}

bool Stack::isEmpty() {           //判断栈内是否为空
    if (top == 0) {
        return true;
    }
    else {
        return false;
    }
}

int len;
double num1[100] = { 0 };
int len1 = 0;

void change(double a[], int n) {     //中缀换前缀
    Stack idx(n);                     //放数据的栈
    Stack symbol(n);                  //放符号的栈
    for (int i = n - 1; i >= 0; i--) {
        if (a[i] < 0) {                 //小于0说明遇到了符号
            int flag = 0;
            char temp = a[i] + '0';
            if (temp == '+' || temp == '-') {
                flag = 1;
                while (flag == 1) {
                    if ((symbol.pop() + '0') == '*' || (symbol.pop() + '0') == '/') {  //比较符号优先级
                        idx.push(symbol.getTop());
                    }
                    else {
                        flag = 0;
                        symbol.push(a[i]);
                    }
                }
            }
            else if (temp == '(') {
                while (symbol.pop() != (')' - '0')) {        //当遇到右括号时,将symbol栈里面的元素放到idx栈
                    idx.push(symbol.getTop());               //此步骤一直持续到遇到symbol栈内的下一个左括号为止
                }                                            //或者持续到栈内空
                symbol.getTop();                             //将左括号出栈
            }
            else {
                symbol.push(a[i]);                           //入栈
            }
        }
        else {
            idx.push(a[i]);
        }
    }
    while (symbol.isEmpty() == false) {
        idx.push(symbol.getTop());
    }
    while (idx.isEmpty() == false) {
        num1[len1] = idx.getTop();
        len1++;
    }
}

double calculate(double a[], int n) {                   //波兰表达式计算,n为数据和符号的个数
    Stack s(n);
    for (int i = n - 1; i >= 0; i--) {
        if (a[i] < 0) {
            char flag = a[i] + '0';
            double b = s.getTop();
            double c = s.getTop();
            double m = 0;
            if (flag == '+') {
                m = b + c;
            }
            else if (flag == '-') {
                m = b - c;
            }
            else if (flag == '*') {
                m = b * c;
            }

            else if (flag == '/') {
                m = b / c;
            }
            s.push(m);
        }
        else {
            s.push(a[i]);
        }
    }
    return s.getTop();
}

int main()
{
    char b;
    char c[20] = { '\0' };
    double num[100] = { 0 };
    int k = 0, len = 0;
    int flag;
    cout << "请输入要计算的中缀表达式(四则混合运算式):";
    while (true) {     //第一步,将表达式字符串转为double型的数组
        scanf("%c", &b);
        if (b == '\n') {
            if (flag == 1) {
                num[len++] = atof(c);
            }
            break;
        }
        if (b >= '0' && b <= '9' || b == '.') {
            c[k++] = b;
            flag = 1;
        }
        else if (b < '0' && flag == 1) {
            num[len++] = atof(c);
            flag = 0;
            for (int i = 0; i < k; i++) {         //清空字符串c
                c[i] = 0;
            }
            k = 0;
        }
        if (b < '0' && flag == 0) {
            num[len++] = b - '0';
        }
    }
    cout << "将中缀表达式转为double数组,便于储存、计算:";
    for (int i = 0;i < len;i++) {
        cout << num[i] << ' ';
    }
    cout << endl;
    change(num, len);       //第二步,中缀表达式转为前缀表达式
    cout << "将中缀表达式转为前缀表达式:";
    for (int i = 0;i < len1;i++) {
        cout << num1[i] << ' ';
    }
    cout << endl;
    cout << "计算结果为:";
    cout << calculate(num1, len1) << endl;  //第三步,计算前缀表达式
    return 0;
}

这是代码运行的结果,正数,小数都可以运算。
在这里插入图片描述
在此分享出自己的学习经历,希望大家能指出上文代码的不足,也希望可以帮助到后面学习的人。

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

诚勤的桃某光

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值