算法练习-逆波兰表达式求值(思路+流程图+代码)

news2025/1/11 14:50:15

难度参考

        难度:中等

        分类:栈与队列

        难度与分类由我所参与的培训课程提供,但需要注意的是,难度与分类仅供参考。且所在课程未提供测试平台,故实现代码主要为自行测试的那种,以下内容均为个人笔记,旨在督促自己认真学习。

题目     

        根据逆波兰表示法,求表达式的值。
        有效的运算符包括+,·,*,/。每个运算对像可以是整数,也可以是另一个逆波兰表达式。

        说明:

        整数除法只保留整数部分。给定逆波兰表达式总是有效的。换句话说,表达式总会得出有效数值且不存在除数
为0的情况。

        示例1:

        输入:["2","1","+","3","*门
        输出:9

        解释:该算式转化为常见的中缀算术表达式为:(2+1)*3)=9

        示例2:
        输入:["4","13","5","/","+"门

        输出:6
        解释:该算式转化为常见的中缀算术表达式为:(4+(13/5)=6

思路

        逆波兰表达式求解的一般思路是使用栈来存储操作数,然后遍历逆波兰表达式的每个元素,根据遇到的操作符进行相应的计算,并将结果重新入栈。遍历逆波兰表达式,并根据运算符进行相应的计算操作。

        具体步骤如下:

  1. 定义一个栈,用于存储中间结果。
  2. 遍历逆波兰表达式,对于每个元素:
    • 如果是数字,则将其转换为整数并压入栈中。
    • 如果是运算符,则从栈中取出两个数字进行相应的运算。将运算结果压入栈中。
  3. 遍历结束后,栈中存储的元素即为表达式的最终结果。

示例

        逆波兰表达式是一种不需要括号的数学表达式表示法,操作符位于操作数之后。

        主要思路是使用一个栈来存储操作数,并依次遍历输入的表达式。当遇到操作符时,从栈中弹出对应数量的操作数,并根据操作符进行相应的运算,将运算结果压入栈中。当遍历完整个表达式后,栈中只会剩下一个元素,即表达式的最终结果。

        举个例子来说明,假设我们要求解的逆波兰表达式是:2 1 + 3 *。我们依次遍历表达式中的每个字符:

  • 第一个字符 “2” 是一个操作数,我们直接将其压入栈中;
    "2", "1", "+", "3", "*"  //字符
     ^
    
    st={2}                    //数字
  • 第二个字符 “1” 是另一个操作数,同样将其压入栈中;
    "2", "1", "+", "3", "*"  //字符
          ^
    
    st={2, 1}                    //数字
  • 第三个字符 “+” 是一个加法操作符,此时栈中的顶部两个元素分别是 1 和 2,我们将它们弹出并做加法运算得到 3,将结果 3 压入栈中;
    "2", "1", "+", "3", "*"  //字符
               ^
    
    st={3}                    //数字
  • 第四个字符 “3” 是一个操作数,将其压入栈中;
    "2", "1", "+", "3", "*"  //字符
                    ^
    
    st={3, 3}                    //数字
  • 最后一个字符 “*” 是一个乘法操作符,此时栈中的顶部两个元素分别是 3 和 3,弹出并做乘法运算得到 9,将结果 9 压入栈中。
    "2", "1", "+", "3", "*"  //字符
                    ^
    
    st={9}                    //数字

        最终栈中只剩下一个元素 9,即为表达式的结果。

梳理

        根据逆波兰表达式的性质,每当遇到一个操作符时,它前面的两个操作数已经被处理过并保存在栈中。所以,我们只需要从栈中弹出这两个操作数,根据操作符进行相应的运算,并将结果再次压入栈中。

  • 没有括号:逆波兰表达式的特点是不需要括号来标识优先级,因为操作数和操作符的顺序已经明确,不存在歧义。
  • 简化运算符的处理:由于操作符总是位于操作数之后,栈可以很方便地保存操作数。每当遇到一个操作符,只需要从栈中弹出所需的操作数进行运算,而不需要关心操作数之间的顺序或优先级。
  • 遍历一次求解:由于逆波兰表达式的特点,我们只需要遍历一次表达式即可求解,无需进行多次迭代或递归。

        总体来说,通过利用栈的先入后出(LIFO)的特性,将逆波兰表达式转化为了一种线性的、遍历一次即可求解的算法,从而实现了逆波兰表达式的求值。

代码

#include <iostream> // 包含输入输出流库,用于标准输入输出
#include <stack> // 包含栈库,用于存储操作数
#include <vector> // 包含向量库,用于存储输入的逆波兰表达式
#include <string> // 包含字符串库,用于操作字符串
#include <cstdlib> // 包含标准库,用于字符串转整数的函数

using namespace std; // 使用标准命名空间

int evalRPN(vector<string>& tokens) { // 定义了一个函数,用于计算逆波兰表达式的值,参数为存储表达式的向量
    stack<int> st; // 创建一个整型栈对象,用于存储操作数
    for (string& token : tokens) { // 遍历逆波兰表达式的每个元素
        if (token == "+") { // 如果当前元素为加号
            int num2 = st.top(); // 取出栈顶元素作为第二个操作数
            st.pop(); // 弹出栈顶元素
            int num1 = st.top(); // 取出新的栈顶元素作为第一个操作数
            st.pop(); // 弹出栈顶元素
            st.push(num1 + num2); // 将计算结果入栈
        } else if (token == "-") { // 如果当前元素为减号,逻辑同上
            int num2 = st.top();
            st.pop();
            int num1 = st.top();
            st.pop();
            st.push(num1 - num2);
        } else if (token == "*") { // 如果当前元素为乘号,逻辑同上
            int num2 = st.top();
            st.pop();
            int num1 = st.top();
            st.pop();
            st.push(num1 * num2);
        } else if (token == "/") { // 如果当前元素为除号,逻辑同上
            int num2 = st.top();
            st.pop();
            int num1 = st.top();
            st.pop();
            st.push(num1 / num2);
        } else { // 如果当前元素为数字
            st.push(stoi(token)); // 将字符串转换为整数后入栈
        }
    }
    return st.top(); // 返回栈中最终的结果
}

int main() { // 主函数
    vector<string> tokens = {"2", "1", "+", "3", "*"}; // 创建一个存储逆波兰表达式的向量,其中元素为字符串
    int result = evalRPN(tokens); // 调用 evalRPN 函数计算表达式的值
    cout << result << endl; // 输出计算结果

    tokens = {"4", "13", "5", "/", "+"}; // 更新逆波兰表达式
    result = evalRPN(tokens); // 调用 evalRPN 函数计算表达式的值
    cout << result << endl; // 输出计算结果

    return 0; // 返回 0,表示正常结束程序
}

打卡

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

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

相关文章

能耗在线监测系统在节能管理中的应用

上海安科瑞电气股份有限公司 胡冠楠 咨询家&#xff1a;“Acrelhgn”&#xff0c;了解更多产品资讯 摘要&#xff1a;开展能耗在线监测系统建设&#xff0c;对加强政府部门和企业节能管理中的应用前景&#xff0c;分析系统在能源消费预测分析、能效对标、节能监察、能源精细化…

【Oracle云】OCI DevOps Services 构建自动化流水线 (1) - 基础架构流程 OCI 代码仓库使用

OCI DevOps Services 是 Oracle Cloud Infrastructure (OCI) 提供的一项独立的 CI/CD 服务&#xff0c;旨在支持用户构建自动化的流水线&#xff0c;实现更高效、可靠的软件交付。在本系列的第一篇文章中&#xff0c;我们将深入探讨 OCI DevOps Services 的基础架构流程&#x…

探索设计模式的魅力:精准解读桥接模式-用桥接模式构建可扩展的软件系统

设计模式专栏&#xff1a;http://t.csdnimg.cn/nolNS 目录 一、了解桥接模式&#xff1a;探索抽象和实现的分离 1.1 开-闭原则 1.2 组合/聚合复用原则 1.3 定义 1.4 用意 1.5 基本思想 1.6 组成部分 1.7 桥梁模式的示意性系统的结构图 二、桥接模式的优势&#xff1a…

RK356X RKAndroid12 TF卡配置 自动挂载

RK356X RKAndroid12 TF卡配置 自动挂载 RK3568 有三个SDMMC接口&#xff0c;分别为SDMMC0 SDMMC1 SDMMC2 DTS 配置 1. max-frequency <150000000>; 此配置设置 SD 卡的运行频率&#xff0c;虽然设置为 150M &#xff0c;但是还要根据 SD 卡的不同模式进行调整。这…

C语言系列-浮点数在内存中的存储

&#x1f308;个人主页: 会编程的果子君 ​&#x1f4ab;个人格言:“成为自己未来的主人~” 目录 浮点数在内存中的存储 浮点数的存储 浮点数存的过程 浮点数取的过程 题目解析 浮点数在内存中的存储 常见的浮点数&#xff1a;3.14159.1E10等&#xff0c;浮点数家族包括&…

使用websocket后端接入文心一言

最近再写项目练手&#xff0c;想着最近大模型那么火&#xff0c;也想接入项目来玩一玩&#xff0c;于是去了解了一下相关的api和通信协议&#xff0c;最后选择了文心一言进行集成&#xff0c;国内的相对稳定。ERNIE-Bot-turbo - 千帆大模型平台 | 百度智能云文档 (baidu.com) …

elementUI实现selecttree自定义下拉框树形组件支持多选和搜索

elementUI实现selecttree自定义下拉框树形组件支持多选和搜索 效果图定义子组件父组件应用 效果图 定义子组件 主要结合el-select和el-tree两个组件改造的。 <template><div class"selectTree"><el-select filterable :filter-method"filterMe…

AI学习(4): PyTorch实战-手写数字识别

1.介绍 在之前的文章中介绍了PyTorch的环境安装&#xff0c;和张量(tensor)的基本使用&#xff0c;为防止陷入枯燥的理论学习中&#xff0c;在这篇文章&#xff0c;我们将进行项目实战学习&#xff0c;项目主要内容: 基于MNIST数据集&#xff0c;实现一个手写数字识别的神经网…

基于OpenCV的高压电力检测项目案例

一、项目背景与目标 随着高压电力设施的日益增多&#xff0c;传统的巡检方式已无法满足现代电力系统的需求。为此&#xff0c;我们决定利用计算机视觉技术&#xff0c;特别是OpenCV库&#xff0c;开发一个高压电力检测系统。目标是实现自动化、高精度的电力设备检测&#xff0c…

《动手学深度学习(PyTorch版)》笔记4.8

注&#xff1a;书中对代码的讲解并不详细&#xff0c;本文对很多细节做了详细注释。另外&#xff0c;书上的源代码是在Jupyter Notebook上运行的&#xff0c;较为分散&#xff0c;本文将代码集中起来&#xff0c;并加以完善&#xff0c;全部用vscode在python 3.9.18下测试通过。…

get out of black background

文章目录 基础 Sequence settings (after selected a Sequence) 看见 ( 让Pr表示透明 ) Effects-> Color Key, drag into your Sequence >.如果看不到 Effects 面板, 可以在 Window 菜单中打开 在Effect Controls 你可以调整 Color Key 的效果了先吸取黑色 还可以使用ma…

Python笔记(二)—— Python判断语句

2.1 布尔类型和比较运算符 布尔类型用于表示&#xff1a;真和假 比较运算符用于计算&#xff1a;真和假 1. 布尔&#xff08;bool&#xff09;表示现实生活中的逻辑&#xff0c;即真和假 True表示真False表示假 True本质上是一个数字记作1&#xff0c;False记作0 定义变…

17.Golang channel的基本定义及使用

目录 概述实践无缓冲 channel代码结果 缓冲 channel代码结果 channel的关闭特点代码结果range代码结果 select channel代码结果 结束 概述 此篇文章介绍 channel 的用法 无缓冲 channel缓冲 channelchannel的关闭特点range channelselect channel 每一种&#xff0c;配上完整…

Flutter 和 Android原生(Activity、Fragment)相互跳转、传参

前言 本文主要讲解 Flutter 和 Android原生之间&#xff0c;页面相互跳转、传参&#xff0c; 但其中用到了两端相互通信的知识&#xff0c;非常建议先看完这篇 讲解通信的文章&#xff1a; Flutter 与 Android原生 相互通信&#xff1a;BasicMessageChannel、MethodChannel、…

075:vue+mapbox 利用高德地址逆转换,点击地图,弹出地址信息

第075个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+mapbox中利用高德逆地理编码,点击地图,弹出某点坐标和地址信息。这里要仔细阅读高德地图的逆编码API,同时要注意的是,这种转换在中国很好用,到了欧美国家就不好使了。同时这个底图是天地图的图像和标记。 直接…

如何在Raspberry Pi上启用SSH并结合cpolar内网穿透实现公网远程访问本地树莓派

文章目录 如何通过 SSH 连接到树莓派步骤1. 在 Raspberry Pi 上启用 SSH步骤2. 查找树莓派的 IP 地址步骤3. SSH 到你的树莓派步骤 4. 在任何地点访问家中的树莓派4.1 安装 Cpolar4.2 cpolar进行token认证4.3 配置cpolar服务开机自启动4.4 查看映射到公网的隧道地址4.5 ssh公网…

React16源码: React中处理hydrate的核心流程源码实现

hydrate 1 &#xff09;概述 hydrate 在react当中不算特别重要, 但是很多时候会用到的一个API这个 API 它主要作用就是在进入第一次渲染的时候&#xff0c;如果本身 dom 树上面已经有一个dom结构存在是否可以去利用这一部分已经存在的dom&#xff0c;然后去避免掉在第一次渲染…

全国疫情实时监测系统(附源码)

目录 一.项目背景 1.有力支持疫情防控知识传播 2.迅速锁定“涉疫”人员流动轨迹 3.开展疫情发展态势预测与溯源 4.一图胜过千言万语&#xff01;&#xff01;&#xff01; 二.研究过程&#xff08;项目技术的利用&#xff09; 1.总述 2.所用技术介绍 2.1Python 2.2Pyt…

免费的ChatGPT网站(7个)

还在为找免费的chatGPT网站或者应用而烦恼吗&#xff1f;博主归纳总结了7个国内非常好用&#xff0c;而且免费的chatGPT网站&#xff0c;AI语言大模型&#xff0c;我们都来接触一下吧。 免费&#xff01;免费&#xff01;免费&#xff01;...&#xff0c;建议收藏保存。 1&…

TPH-YOLOv5:基于Transformer预测头改进的YOLOv5开发构建麦穗检测计数分析系统

关于小麦麦穗或者是麦粒相关的开发实践不多&#xff0c;但前文也有所涉及&#xff0c;感兴趣的话可以自行移步阅读即可&#xff1a; 《基于轻量级yolov5nCBAM开发构建全球小麦麦穗智能检测计数系统》 《基于YOLOv5[n/s/m/l/x]全系列参数模型开发构建小麦麦穗颗粒智能化精准检…