【算法练习Day10】有效的括号删除字符串中的所有相邻重复项逆波兰表达式求值

news2025/1/15 20:39:28

在这里插入图片描述

​📝个人主页:@Sherry的成长之路
🏠学习社区:Sherry的成长之路(个人社区)
📖专栏链接:练题
🎯长路漫漫浩浩,万事皆有期待

文章目录

  • 有效的括号
  • 删除字符串中的所有相邻重复项
  • 逆波兰表达式求值
  • 总结:

有效的括号

20. 有效的括号

在这里插入图片描述

这道题相信学过数据结构的同学应该并不陌生,这道题是一道在学习栈的时候的一道十分经典的题型,起初第一次接触的时候,多少会有点感觉难,现在做来还行的。

题目要求就是将所有的括号匹配起来,主要有两种情况会匹配失败:
一种是左括号或者右括号出现多余的情况,另一种情况是前一个括号与后一个括号不能连续匹配,我们可以根据栈这个数据结构的特点,使左括号字符都放进去,遇到右括号的时候,top一下看看栈顶元素是否为相匹配的左括号,如果相匹配我们将其弹出栈,然后接着遍历,如果不匹配我们直接return false

注意:如果在碰到右括号时候栈为空则说明栈中没有左括号,那么此时直接弹出,说明右括号有多余的。如果将整个字符都遍历完了,依然没有return,那么则说明第二种情况已经判断完毕,并没有括号不匹配的情况出现,但是此时并不代表就一定没有错误,当匹配完毕之后如果栈中还剩下括号说明左括号有多余的。

class Solution {
public:
    bool isValid(string s) 
    {
        stack<int> arr1;
        for(int i=0;i<s.size();i++)
        {
            if(s[i]!=')'&&s[i]!='}'&&s[i]!=']')
            {
                arr1.push(s[i]);
            }
            if(s[i]==')')
            {
                if(arr1.empty()==true||arr1.top()!='(')
                {
                    return false;
                }
                else
                {
                    arr1.pop();
                }
            }
            if(s[i]=='}')
            {
                if(arr1.empty()==true||arr1.top()!='{')
                {
                    return false;
                }
                else
                {
                    arr1.pop();
                }
            }
            if(s[i]==']')
            {
                if(arr1.empty()==true||arr1.top()!='[')
                {
                    return false;
                }
                else
                {
                    arr1.pop();
                }
            }
            
        }
        if(arr1.empty()==true)
        {
            return true;
        }
        else
        {
            return false;
        }
    }
};

还有另一种思路:在遇到左括号时候往栈里直接加入右括号,然后碰到右括号时候在进行判断,和上一种思路基本上是一样的。

class Solution {
public:
    bool isValid(string s) {
        if(s.size()%2!=0)
        {// 如果s的长度为奇数,一定不符合要求
            return false;
        }
        stack<char>st;
        for(int i=0;i<s.size();i++)
        {
            if(s[i]=='(')
            {
                st.push(')');
            }
            else if(s[i]=='{')
            {
                st.push('}');
            }
            else if(s[i]=='[')
            {
                st.push(']');
            }
             // 第二种情况:遍历字符串匹配的过程中,发现栈里没有我们要匹配的字符。所以return false
             // 第三种情况:遍历字符串匹配的过程中,栈已经为空了,没有匹配的字符了,说明右括号没有找到对应的左括号 return false
            else if(st.empty()||st.top()!=s[i])
            {
                return false;
            }
            else
            { // st.top() 与 s[i]相等,栈弹出元素
                st.pop();
            }
        }
         // 第一种情况:此时我们已经遍历完了字符串,但是栈不为空,说明有相应的左括号没有右括号来匹配,所以return false,否则就return true
        return st.empty();
    }
};

删除字符串中的所有相邻重复项

1047. 删除字符串中的所有相邻重复项

在这里插入图片描述

这道题是消除字符串中连续的字符,只要有相邻重复就直接删除,所以并不是我们第一眼看到只有一处相邻相同字符就只删一次,可能在删除一处之后又有新的元素碰到一起,这时栈派上了用场,它可以存储我们遍历过的数据,我们只需要判断当前遍历的数据是否和栈顶元素一样,一致就将其弹出,这样不仅解决了表面的相邻相同字符消去,当消去后如果有两个新的相邻相同元素碰到一起了,我们也能发现并消除他们,一举多得。

class Solution {
public:
    string removeDuplicates(string s) {
        stack<int> st;
        for(char s1:s)
        {
            if(st.empty()||s1!=st.top())
            {
                st.push(s1);
            }
            else
            {
                st.pop();// s 与 st.top()相等的情况
            }
        }
        string result="";
        while(!st.empty())
        {// 将栈中元素放到result字符串汇总
            result+=st.top();
            st.pop();
        }
        reverse(result.begin(),result.end()); // 此时字符串需要反转一下
        return result;
    }
};

上面的代码我们用字符串来模拟栈的一个实现,让字符串的头部充当栈底,尾部充当栈顶,这样做的好处是避免了直接使用栈的时候数据元素是倒序且还要转换为字符串输出。

class Solution {
public:
    string removeDuplicates(string s) {
        string result;
        for(char s1:s)
        {
            if(result.empty()||result.back()!=s1)
            {
                result.push_back(s1);
            }
            else
            {
                result.pop_back();
            }
        }
        return result;
    }
};

逆波兰表达式求值

150. 逆波兰表达式求值
在这里插入图片描述

逆波兰表达式求值,这道题的难度是中等题,并不是是它有多么难的思路,而是如果你没有接触到这个题,不是很容易想到解题思路,逆波兰表达式,实际上就是一种后缀表达式,可以把我们日常熟悉的一个等式(1+2)* (3*4)看成中序表达式,画树形图很容易得出,那么后续表达式大题运算思路就是两个数遇到符号再进行运算操作,没遇到符号时候先保留起来,每次遇到符号拿出两个操作数操作,这种思路很适用于栈的数据结构。

class Solution {
public:
    int evalRPN(vector<string>& tokens) {
        stack<int> st;
        for(int i=0;i<tokens.size();i++)
        {
            if(tokens[i]=="+"||tokens[i]=="-"||tokens[i]=="*"||tokens[i]=="/")
            {
                long long num1=st.top();
                st.pop();
                long long num2=st.top();
                st.pop();
                if(tokens[i]=="+")
                {
                    st.push(num1+num2);
                }
                if(tokens[i]=="-")
                {
                    st.push(num2-num1);
                }
                if(tokens[i]=="*")
                {
                    st.push(num2*num1);
                }
                if(tokens[i]=="/")
                {
                    st.push(num2/num1);
                }
            }
            else
            {
                st.push(stoll(tokens[i]));//在C++中,std::stoll函数用于将字符串转换为长整型(long long)数据类型。
            }
        }
        int result= st.top();
        st.pop();// 把栈里最后一个元素弹出(其实不弹出也没事)
        return result;
    }
};

具体代码如上,在未碰到操作符时,将数字入栈,遇到操作符再取出两个操作符,值得注意的是操作数顺序很重要是第二个操作数相对于第一个操作数来进行操作,数据放入里的stoll函数,作用是将字符串转换为long数据类型。此外这道题并不需要我们来判断传输进来的数据是否合法,只需要我们写出判断字符和数据的代码即可。

总结:

今天我们完成了有效的括号、删除字符串中的所有相邻重复项、逆波兰表达式求值三道题目,相关的思想需要多复习回顾。接下来,我们继续进行算法练习·。希望我的文章和讲解能对大家的学习提供一些帮助。

当然,本文仍有许多不足之处,欢迎各位小伙伴们随时私信交流、批评指正!我们下期见~

在这里插入图片描述

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

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

相关文章

树莓派4B与STM32串口通信

目录 2上篇文章的补充 2.1 树莓派通信设置 3树莓派与STM32通信 3.1接线准备 3.2代码 3.2.1 STM32代码&#xff1a; 3.2.2树莓派代码&#xff1a; 2上篇文章的补充 2.1 树莓派通信设置 在上篇文章的基础上&#xff0c;进一步的设置 终端输入&#xff1a;sudo minicom …

【Golang】接口 IO操作

文章目录 1、接口1-1、为什么要有接口1-2、接口的定义1-3、接口的实现条件1-4、类型与接口的关系1-5、空接口 2、IO操作2-1、Reader2-2、Writer2-3、bufio2-4、ioutil工具包2-5、实现一个cat命令 1、接口 1-1、为什么要有接口 在Go语言中接口&#xff08;interface&#xff0…

Flink--7、窗口(窗口的概念、分类、API、分配器、窗口函数)、触发器、移除器

星光下的赶路人star的个人主页 内心的平静始于不再让他人掌控你的感情 文章目录 0、前言1、窗口&#xff08;Window&#xff09;1.1 窗口的概念1.2 窗口的分类1.3 窗口API概览1.4 窗口分配器&#xff08;Window Assigner&#xff09;1.4.1 时间窗口1.4.2 计数窗口 1.5 窗口函数…

什么是Vue的JSX语法?如何使用JSX语法

Vue的JSX语法&#xff1a;更接近JavaScript的模板语言 Vue.js是一个流行的JavaScript框架&#xff0c;用于构建交互式的Web应用程序。虽然Vue通常使用模板语法来构建用户界面&#xff0c;但它也提供了JSX语法的支持&#xff0c;使开发人员能够更接近JavaScript的表达方式来构建…

Eclipse MAT解析headp dump,total size小于file size

1. 问题描述 使用Eclipse MAT分析20GB的heap dump文件 最后解析出来dump size只有1GB 2. 原因&#xff1a;heap dump中包含许多unreachable objects Eclipse MAT的官方文档&#xff0c;《Basic Tutorial》章节&#xff0c;有对上图的Overview page做介绍 针对total size小…

Open Cascade旋转变换平行线

在本人开发的弯管自动CAM软件中&#xff0c;有一个问题一直没有解决&#xff0c;就是180度平行管路需要做角度微调&#xff0c;以便进行YBC预览。研究了一番后&#xff0c;搞定了这个问题&#xff0c;关键在于采用OCC库实现拓扑变换。 本文将介绍如何使用OpenCASCADE库来实现平…

微信公众号如何修改主体?

公众号账号迁移的作用是什么&#xff1f;只能变更主体吗&#xff1f;1.可合并多个公众号的粉丝、文章&#xff0c;打造超级大V2.可变更公众号主体&#xff0c;更改公众号名称&#xff0c;变更公众号类型——订阅号、服务号随意切换3.可以增加留言功能4.个人订阅号可迁移到企业名…

牛客题霸 -- 【模板】完全背包

参考代码&#xff1a; 未优化的代码&#xff1a; int n; int V; const int N1010; int v[N]; int w[N]; int dp[N][N];int main() {cin>>n>>V;for(int i1;i<n;i){cin>>v[i]>>w[i];}//第一问&#xff1a;//dp表中的第一行全是0&#xff0c;无需初始…

【初识Linux】上

初识Linux上 一、Linux背景1.1 UNIX发展的历史1.2 UNIX发展的历史 二、开源三、官网Linux官网 四、企业应用现状五、发行版本六、 os概念&#xff0c;定位 本博客简介 初始Linux操作系统初识shell命令 ,了解若干背景知识。使用常用Linux命令了解Linux权限概念与思想,能深度理解…

Centos7配置NAT网络

1、在网上查了好多内容&#xff0c;配置始终不能ping www.baidu.com&#xff0c;搞了一下午还是一样。 2、晚上查看DHCP配置&#xff0c;看到子网ip是192.168.70.0&#xff0c;但是起始ip为128起&#xff0c;于是将/etc/sysconfig/network-scripts/ifcfg-ens33 文件的ip换成13…

凉鞋的 Godot 笔记 104. 测试所涉及的窗口

104. 测试所涉及的窗口 在上一篇&#xff0c;笔者简单介绍了检视器窗口&#xff0c;如图所示&#xff1a; 我们接着介绍上图中的最后一个部分内容&#xff0c;测试部分。 测试部分我们只做了一件非常简单的操作&#xff0c;就是点击了一下运行当前场景按钮&#xff0c;查看结…

多目标黏菌算法(MOSMA)附带多个多目标性能指标

1 黏菌算法 http://t.csdnimg.cn/yArV5 2 多目标黏菌算法 %% Multiple Objective Slime Mould Algorithm (MOSMA) clc clear all D 30; % Number of decision variables M 2; % Number of objective functions KMD; LB ones(1, D).*0; % LB - A vector of decimal value…

PAT 1048 数字加密

PAT 1048 数字加密 题目描述思路讲解代码展示 题目描述 思路讲解 分析&#xff1a;首先将a和b倒置&#xff0c;将字符串a和b中较短的那个末尾添加0直到两个字符串长度相等&#xff0c;然后从0开始依次处理每一位&#xff0c;如果当前位是奇数位&#xff08;i % 2 0&#xff0…

力扣第 365 场周赛虚拟参赛

有序三元组中的最大值 I class Solution { public:long long maximumTripletValue(vector<int>& nums) {vector<long long> num;for (auto &item:nums) {num.push_back(item*1ll);}long long z 0,f 1000000;long long ans 0;long long maxx num[0],mi…

凉鞋的 Godot 笔记 105. 第一个通识:编辑-测试 循环

105. 第一个通识&#xff1a;编辑-测试 循环 在这一篇&#xff0c;我们简单聊聊此教程中所涉及的一个非常重要的概念&#xff1a;循环。 我们在做任何事情都离不开某种循环&#xff0c;比如每天的 24 小时循环&#xff0c;一日三餐循环&#xff0c;清醒-睡觉循环。 在学习一…

【SLAM数学基础】李群与李代数 BCH近似公式

三维旋转构成了三维旋转群 SO(3)&#xff0c;其对应的李代数为 s o \mathfrak{so} so(3)&#xff1b;三维变换构成了三维变换群 SE(3)&#xff0c;其对应的李代数为 s e \mathfrak{se} se(3)。 1.指数映射 李代数元素到李群元素的映射为指数映射&#xff0c;其中 s o \mat…

代码随想录算法训练营第五十五天 | 动态规划 part 12 | 300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组

目录 300.最长递增子序列思路代码 674. 最长连续递增序列思路代码 718. 最长重复子数组思路代码 300.最长递增子序列 Leetcode 思路 dp[i]表示i之前包括i的以nums[i]结尾的最长递增子序列的长度递推公式&#xff1a;if (nums[i] > nums[j]) dp[i] max(dp[i], dp[j] 1)初…

力扣:118. 杨辉三角(Python3)

题目&#xff1a; 给定一个非负整数 numRows&#xff0c;生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中&#xff0c;每个数是它左上方和右上方的数的和。 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;力扣&#xff08;LeetCode&#xff09;官…

linux 笔记:远程服务器登录jupyter notebook

1 生成jupyter notebook 配置文件&#xff08;服务器端&#xff09; jupyter notebook --generate-config #Writing default config to: /home/shuailiu/.jupyter/jupyter_notebook_config.py2 Ipython中设置密码&#xff08;服务器端&#xff09; 3 修改jupyter 配置文件&…

QT实现TCP服务器客户端搭建的代码,现象

1.效果 2.服务器&#xff1a; 2.1&#xff1a;ui界面 2.2&#xff1a;头文件 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> #include <QTcpSocket> #include <QList> #include <QMessageBox> #include <QDeb…