c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。

2024-03-29 06:08

本文主要是介绍c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

      最近在学习数据结构与算法,学到栈这里,就基于栈实现了一个简答四则表达式算值的程序。平时我们写的那
种表达式就是中缀,而计算机处理中缀是不占优势的一般都是将中缀转成后缀再计算值,在这里我也利用这个思路,
将中缀表达式分为以下两部:
     A.中缀表达式转成后缀表达式,主要有以下几步:
 
a.初始化一个运算符栈  b.从左到右读取一个字符  c.如果是操作数就直接输出到后缀表达式  d.如果是左括号就压入操作符栈  e.如果是右括号,则把与之匹配的左括号之间的运算符全部输出  f.如果是运算符就分情况执行以下操作  1).如果栈为空就直接压入  2).如果栈中有运算符,切当前运算符比栈顶运算符优先级高就压入,否则就弹出栈顶运算符  ,并一直重复1)、2)直到栈顶运算符优先级低于待处理运算符或栈为空  3).重复b直到读取完毕 B.后缀表达式求值,这个就相对简单一些了: a.初始化一个操作数栈 b.依次读取后缀表达式,遇到运算数就压进运算数栈,遇到操作符就将栈顶两个数pop出来求值 然后push进栈。一直循环执行b知道后缀表达式读取完毕,最后操作数栈顶那个数就是表达式。 

下面就直接贴代码了,代码注释基本能够帮助快速读懂程序。

#include <iostream>
#include <stack>
using namespace std;int getPriority(char c);    //输入运算符返回优先级
bool isLeft(char c);        //判断是否是左括号
bool isRight(char c);       //判断是否是右括号
bool isOperator(char c);    //判断是否是操作符(+-*/)
bool isNumber(char c);      //判断是否是数字
double calculate(string postfix);   //传入后缀表达式,计算值
string convertToPostfix(const string& expression); //将中缀表达式转化为后缀表达式int main() {stack<char> operatorStack;string expression,postfix;cout << "请输入要计算的表达式:"<<endl;getline(cin,expression);postfix = convertToPostfix(expression);//输出结果cout << "后缀表达式为(数字之间以#号分割)->"<<postfix << endl;if (expression.at(expression.length()-1) == '=')cout <<expression<<calculate(postfix)<<endl;elsecout <<expression<<"="<<calculate(postfix)<<endl;
}string convertToPostfix(const string& expression){stack<char> operatorStack;string postfix;//依次读取输入的字符并按字符类型分别处理for(int current = 0; current < expression.length(); current++){char c = expression.at(current);if(!isNumber(c) && !isOperator(c) && c !='(' && c != ')'){cout << "请输入正确表达式!" << endl;exit(0);}if(c != ' ' && c != '='){//如果是左括号就压栈if(isLeft(c)) {operatorStack.push(c);} else if (isRight(c)){     //是右括号则将和左括号之间的pop出去拼接到后缀表达式while(!operatorStack.empty() && operatorStack.top() != '('){postfix.append(1,operatorStack.top());operatorStack.pop();}operatorStack.pop();        //最后将左括号pop} else if(isOperator(c)){   //如果是操作符if(current+1 < expression.length()){if (isOperator(expression.at(current+1))) {cout << "请输入正确的表达式";exit(0);}}if(current-1 >= 0){if(isOperator(expression.at(current-1))){cout << "请输入正确的表达式";exit(0);}}if(operatorStack.empty()){  //如果操作符栈为空就直接压栈operatorStack.push(c);} else{                     //否则就将栈顶优先级低的依次出栈,直到栈为空或者,栈顶运算符优先级低于当前运算符while(getPriority(c) <= getPriority(operatorStack.top())){postfix.append(1,operatorStack.top());operatorStack.pop();if(operatorStack.empty())break;}operatorStack.push(c);}} else {    //最后如果是数字的话就直接接到后缀表达式postfix.append(1,c);            //同时为了方便后面计算后缀表达式值的时候明确多位数的起止,特在每个整数后面加一个分隔符$if(current+1 < expression.length()){if (!isNumber(expression.at(current+1))){postfix.append(1,'#');//数字分隔符}}}}}//中缀表达式读取完毕后将运算符栈中剩下的运算符拼接到后缀表达式中while(!operatorStack.empty()){if(operatorStack.top() == '(' || operatorStack.top() == ')'){cout << "左右括号不匹配,表达式错误!";exit(0);}postfix.append(1,operatorStack.top());operatorStack.pop();}return postfix;
}
//返回运算符优先级
int getPriority(char c){if(c == '*' || c == '/')return 2;else if(c == '+' || c == '-')return 1;elsereturn  0;
}bool isRight(char c){if(c == ')')return true;elsereturn false;
}bool isLeft(char c){if(c == '(')return true;elsereturn false;
}bool isOperator(char c){if(c == '*' || c == '/' || c == '+' ||c == '-')return true;elsereturn false;
}double calculate(string postfix){stack<double> result;char c;for (int i = 0; i < postfix.length();i++) {c = postfix.at(i);if (isOperator(c)) {double a = result.top();result.pop();double b = result.top();result.pop();double temp;switch (c) {case '+' :temp = b + a;break;case '-' :temp = b - a;break;case '/' :if(a == 0){cout << "除零错误!";exit(0);}temp = b / a;break;case '*' :temp = b * a;default:break;}result.push(temp);} else if(isNumber(c)){double num = c - '0';while(isNumber(postfix.at(i+1))){num = num*10+ (postfix.at(i+1) - '0');i++;}result.push(num);}}return result.top();
}bool isNumber(char c){if(c>='0'&& c<='9')return true;elsereturn false;
}

当然代码中加入了一些判别表达式错误的片段,比如左右括号不匹配,或是除数为零,以及为了支持多位数,特意在每个
数结束处加一个#号方便后缀表达式计算值时识别,相关代码段如下:

            postfix.append(1,c); //同时为了方便后面计算后缀表达式值的时候明确多位数的起止,特在每个整数后面加一个分隔符#                if(current+1 < expression.length()){if (!isNumber(expression.at(current+1))){postfix.append(1,'#');//数字分隔符}}

 

 

 

 

 

 

 

 

#

 

这篇关于c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/857805

相关文章

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库