中缀表达式转后缀表达式(逆波兰表达式)及如何计算后缀表达式

目录

中缀、后缀表达式简介

中缀转后缀的规则

模拟中缀转后缀 

中缀转后缀代码 

后缀表达式求值

后缀表达式求值代码

Leetcode相关题目 


中缀、后缀表达式简介

首先说说什么是中缀表达式,中缀表达式中,操作符是以中缀形式处于操作数的中间。例如,在表达式“3 + 4”中,“+”是中缀操作符,位于两个操作数“3”和“4”之间。对于我们而言,它最直观。下面说说后缀表达式,

逆波兰表达式(Reverse Polish Notation,RPN),也被称为后缀表达式,是一种算术表达式的表示方法。在这种表示法中,运算符被写在操作数的后面,而不是像常规的中缀表达式那样写在操作数的中间。逆波兰表达式不使用括号来指示运算的优先级,而是完全依赖于运算符出现的顺序以及运算符本身的优先级规则。也就是说,后缀表达式严格遵循从左往右计算。

中缀转后缀的规则

下面是中缀转后缀的思维导图,有一些细节不方便体现,我们到代码实现部分在谈。

模拟中缀转后缀 

根据上面思维导图中的规则,我们来模拟一下中缀转后缀。

希望通过上面的一个简单例子,我们明白了中缀转后缀的规则。上面被框起来的部分,是个细节,我在写代码的时候犯了这样的错误——我在把+出栈后,直接让-入栈,可把我坑惨了。出现错误后,我再次看了一遍代码,重点看了这一部分,但是觉得很对,一点都没错,哎。所以,我错在哪了?为什么不可以直接让-进栈?

我们在思考这个问题的时候不要局限于上面的图,因为照着上面的话感觉我犯的错误很对。

我错就错在我以为栈内就只有+这个运算符,+运算符出栈后,栈为空,所以直接把 - 运算符入栈。但实际上,栈内可能还有其他运算符,而且优先级低于或等于栈顶运算符。此时,应该让栈顶运算符出栈,而不是把 - 直接丢入栈内。也就是说,当前运算符 - 还要继续走前面运算符的逻辑。如果这句话暂时不能理解,没关系,看了代码就明白了。

下面我们上个看起来比较复杂的式子练练手。

其中,绿色的栈是递归时创建的。上面的例子值得好好理解。

中缀转后缀代码 

好了,我就不再废话了,上代码!

#include <iostream>
#include <vector>
#include <stack>
#include <map>
using namespace std;//下面需要比较优先级的高低,由于无法直接使用ASCII比较,故进行一下定义
map<char, int> pro = { {'+', 1}, {'-', 1}, {'*', 2}, {'/', 2} };void toRPN(const string& s, size_t& i, vector<string>& v)
{int n = s.size();stack<char> st;//存储操作符while (i < n){//isdigit为C语言的一个函数,功能是判断字符串中的某个字符是否是数字if (isdigit(s[i])){//细节:数字不一定是1位数,所以需要提取出完整的数字string tmp;while (i < n && isdigit(s[i])) tmp += s[i++];v.push_back(tmp);}else if (s[i] == '('){//递归处理i++;//跳过'('toRPN(s, i, v);}else if (s[i] == ')'){//结束递归while (!st.empty()){v.push_back(string(1, st.top()));//查文档——string的构造函数st.pop();}i++;//跳过')'return;}else{//走到这说明就是操作符了if (st.empty() || pro[st.top()] < pro[s[i]])st.push(s[i++]);else{v.push_back(string(1, st.top()));st.pop();//这里不能让i++,之前解释过了}}}//走到这,说明计算式或子表达式结束了,输出栈中所有运算符while (!st.empty()){v.push_back(string(1, st.top()));st.pop();}
}int main()
{vector<string> v;size_t i = 0;string s = "1+2-(3*4+5)-7";toRPN(s, i, v);for (auto s : v) cout << s << " ";cout << endl;return 0;
}

 到目前,你可能会疑惑会质疑,空格不需要处理吗?或者说有空格的话上面程序会不会崩掉?

空格是需要处理的,上面的函数是在处理完原字符串中的空格后再调用的,因为我不想在该函数内部处理空格,如果你想的话也可以,多加个判断就好了。

除了这个问题,代码中还有一个细节——仔细看看toRPN函数的参数。

我是不是都用了引用?因为我们希望程序在递归时处理的也是相同的字符串,递归时处理的结果也可以影响到上层,这是我们想要的。换句话说,s,i,v,为整个程序共用。

后缀表达式求值

前面在介绍后缀表达式的时候提到,后缀表达式严格按照从左到右计算。在计算后缀表达式时,我们需要借助一个栈(当然你可以用其他数据结构模拟栈)来存储数字。

 1 2 + 3 4 * 5 + - 7 -,以此表达式为例。

从左到右遍历表达式。

1)遇到运算数,直接入栈。

2)遇到操作符,取出栈中的两个操作数运算,运算结果存入中栈中。(细节下面说)

3)重复以上步骤,当遍历完表达式,栈顶元素即为答案。

有些操作符,比如 ‘-’ 和 ‘/’,它们是区分左右操作数的,即a - b 和 b - a 的结果可能不一样。这该怎么办?

栈的性质——后进先出。我们从左往右遍历,一定是左操作数先进栈,右操作数后进栈,所有先出的是右操作数,后出的是左操作数,这样就可以区分左右操作数了。其实,这就是选择栈来存储数据的原因。

后缀表达式求值代码

int calculate(vector<string>& s)
{stack<int> st;for (auto str : s){if (str == "+" || str == "-" || str == "*" || str == "/"){int right = st.top();st.pop();int left = st.top();st.pop();int ans = 0;switch (str[0]){case '+': ans = left + right; break;case '-': ans = left - right; break;case '*': ans = left * right; break;case '/': ans = left / right; break;}st.push(ans);}else{st.push(stoi(str));}}return st.top();
}
int main()
{vector<string> v;size_t i = 0;string s = "1+2-(3*4+5)-7";toRPN(s, i, v);cout << calculate(v) << endl;return 0;
}

Leetcode相关题目 

. - 力扣(LeetCode)

上面这题可以我们本片博客讲到的方法解决,虽然不是最简洁的解法,但是可以练练我们上面讲到的方法。代码我在下面贴一份。

class Solution {
public:int calculate(string s) {vector<string> v;stack<int> st;size_t i = 0;toRPN(s, i, v);for(auto& str : v){if(str == "+" || str == "-" || str == "*" || str == "/"){int right = st.top();st.pop();int left = st.top();st.pop();int ans = 0;switch(str[0]){case '+': ans = left + right; break;case '-': ans = left - right; break;case '*': ans = left * right; break;case '/': ans = left / right; break;}st.push(ans);}else{st.push(stoi(str));}}return st.top();}//中缀转后缀map<char, int> pro = { {'+', 1}, {'-', 1}, {'*', 2}, {'/',2} };void toRPN(const string& s, size_t i, vector<string>& v){int n = s.size();stack<char> st;while(i < n){if(s[i] == ' ') i++;else if(isdigit(s[i])){//提取数字string tmp;while(i < n && isdigit(s[i])) tmp += s[i++];v.push_back(tmp);}else{if(st.empty() || pro[s[i]] > pro[st.top()])st.push(s[i++]);else{v.push_back(string(1, st.top()));st.pop();}}}while(!st.empty()){v.push_back(string(1, st.top()));st.pop();}}
};

本篇文章到这就结束啦~如有错误,请您不吝指出,谢谢!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/882519.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ImportError: DLL load failed while importing _rust: 找不到指定的模块。

ImportError: DLL load failed while importing _rust: 找不到指定的模块。 安装qwen-agent 过程需要安装 cryptography 报错 &#xff1a; 177 WARNING: Failed to collect submodules for ‘cryptography.hazmat.backends.openssl’ because importing ‘cryptography.hazm…

【openwrt-21.02】T750 openwrt 概率出现nat46_ipv4_input+0x90/0x4b4问题分析及解决方案

Openwrt版本 NAME="OpenWrt" VERSION="21.02-SNAPSHOT" ID="openwrt" ID_LIKE="lede openwrt" PRETTY_NAME="OpenWrt 21.02-SNAPSHOT" VERSION_ID="21.02-snapshot" HOME_URL="https://openwrt.org/" …

Linux安装Anaconda和Pytorch

又到了一年一度换环境、换服务器不断折腾的时节了&#xff0c;一通折腾后&#xff0c;重新启动遂做记录。 1. Linux安装Anaconda 1.1 离线安装模式 进入官网https://www.anaconda.com/download/success&#xff0c;如图所示&#xff1a; 选择版本进行下载即可。 1.2 在线w…

基于IP的真实地址生成器

ip-geoaddress-generator 是一个基于 Web 的在线应用程序&#xff0c;能够根据 IP 地址生成真实的随机地址信息。通过多个 API 获取位置数据和随机用户信息&#xff0c;该工具为用户提供了完整的虚拟身份。它由 Next.js 和 Radix UI 构建&#xff0c;具备自动检测当前 IP 地址和…

[Linux网络编程]03-TCP协议

一.TCP协议数据通信的过程 TCP数据报如下&#xff0c;数据报中的标志位双端通信的关键。 三次握手: 1.客户端向服务端发送SYN标志位&#xff0c;请求建立连接&#xff0c;同时发送空包 2.服务端向客户端回发ACK标志位(即确认标志位&#xff0c;任何一端发送数据后都需要另一端…

【VUE】【IOS】【APP】IOS Music APP播放器开发

前言 周末闲来无事&#xff0c;学习了下移动端的一些知识。了解到移动端的一些实现方式&#xff0c;先从最简单的开始。本人没有IOS swift 和Android的开发经验。抱着学习态度从简单的入手&#xff0c;经过了解&#xff0c;本人之前自己用vue的写着玩了几个小项目。看到可以用…

《使用Gin框架构建分布式应用》阅读笔记:p101-p107

《用Gin框架构建分布式应用》学习第7天&#xff0c;p101-p107总结&#xff0c;总计7页。 一、技术总结 1.StatusBadRequest vs StatusInternalServerError 写代码的时候有一个问题&#xff0c;什么时候使用 StatusBadRequest(400错误)&#xff0c;什么时候使用 StatusIntern…

c语言中的%运算和/运算

在C语言中&#xff0c;%运算和/运算分别表示取模运算和除法运算。以下是它们的详细解释和用法&#xff1a; 1. % 运算&#xff08;取模运算&#xff09; 取模运算用于计算两个整数相除后的余数。语法如下&#xff1a; result a % b; a 是被除数。b 是除数。result 是 a 除…

了解CSS Typed OM

CSS Typed OM&#xff08;CSS Typed Object Model&#xff09;是一项前沿的技术&#xff0c;旨在改变我们编写和操作CSS的方式。以下是对CSS Typed OM的详细解析&#xff1a; 一、CSS Typed OM概述 CSS Typed OM是一个包含类型和方法的CSS对象模型&#xff0c;它暴露了作为Ja…

1.2电子商务安全内涵

目录 1 电子商务安全的层次 2 计算机网络安全 3电子商务安全的特点 只有在你生命美丽的时候&#xff0c;世界才是美丽的。 —— 顾城 《顾城哲思录》 1 电子商务安全的层次 安全:主体没有危险的客观状态 电子商务安全是一个广泛的概念&#xff0c;它涉及到电子商务的各个方…

react的state是一张快照

快照的概念 在 React 中&#xff0c;state 是组件状态的表现形式&#xff0c;它是一个组件内部可变的状态数据。React 的官方文档中提到 state 是一个快照&#xff08;snapshot&#xff09;&#xff0c;这个概念主要是指 React 的状态更新机制。 当你调用 setState 方法来更新…

es 全文文本分词查询

ES 对 TEXT 类型的分词 我们可以使用分词器来查看分词的结果&#xff0c;默认的分词器是 standard 如果是其他的可以指定分词器 # 标准的分词 POST _analyze {"text": "Hello World!","analyzer": "standard" }全文检索的分词概念 …

现今 CSS3 最强二维布局系统 Grid 网格布局

深入学习 CSS3 目前最强大的布局系统 Grid 网格布局 Grid 网格布局的基本认识 Grid 网格布局: Grid 布局是一个基于网格的二位布局系统&#xff0c;是目前 CSS 最强的布局系统&#xff0c;它可以同时对列和行进行处理&#xff08;它将网页划分成一个个网格&#xff0c;可以任…

PHP函数$_FILES详解

PHP函数$_FILES详解 在PHP中上传一个文件建一个表单要比ASP中灵活得多。具体的看代码。 <form enctype"multipart/form-data" action"upload.php" method"post"> <input type"hidden" name"MAX_FILE_SIZE" value…

linux基础-学习笔记

文章目录 linux软件安装linux系统部署liunx升级linux常见故障及排查思路概要 1. Linux软件安装 软件包管理:Linux系统通常使用包管理工具(如APT、YUM、DNF等)来简化软件安装和管理。用户可以通过命令行快速安装、卸载和更新软件包。源配置:确保软件源(repository)正确配…

嵌入式入门学习——8基于Protues仿真Arduino+SSD1306液晶显示数字时钟

0 系列文章入口 嵌入式入门学习——0快速入门&#xff0c;Let‘s Do It&#xff01; SSD1306 1 Protues查找SSD1306器件并放置在画布&#xff0c;画好电气连接&#xff08;这里VCC和GND画反了&#xff0c;后面仿真出错我才看见&#xff0c;要是现实硬件估计就烧毁了&#xf…

MongoDB 如何做mapreduce

以下是在MongoDB中使用MapReduce的详细步骤和相关说明&#xff1a; 1. MapReduce的概念 MapReduce是一种用于大规模数据处理的编程模型&#xff0c;它由两个主要阶段组成&#xff1a;Map阶段和Reduce阶段。在MongoDB中&#xff0c;MapReduce操作允许在服务器端对数据进行批量…

数字处理系列

&#xff08;1&#xff09;将数字转化成中文的过滤器 <template><div><p>数字转中文&#xff1a;{{ 110 | numberToChinese }}</p></div></template><script>export default {filters: {numberToChinese(num) {const chineseNums …

【时时三省】(C语言基础)函数介绍strncat

山不在高&#xff0c;有仙则名。水不在深&#xff0c;有龙则灵。 ----CSDN 时时三省 strncat 打印结果是hello wor 跟strcat不同的是他后面可以加一个参数 这个参数就是它可以根据后面的数字 来追加多少个字符 这个如果后面的参数改成10的话 就是打印hello world 不会跟strn…

智能EDA小白从0开始 —— DAY20 OrCAD

以下是对OrCAD和MATLAB两种EDA工具的深入解析&#xff0c;内容扩展至约2220字&#xff1a; OrCAD&#xff1a;电子设计自动化的强大工具 OrCAD&#xff0c;作为电子设计自动化&#xff08;EDA&#xff09;领域的佼佼者&#xff0c;为电子工程师们提供了一套全面的设计解决方案…