面试中考察栈和队列的经典算法题

💝💝💝如果你对顺序表的概念与理解还存在疑惑,欢迎观看我之前的作品👉【栈和列队详解】

上篇文章👉 【面试中顺序表常考的十大题目解析】


目录

💯前言

💯栈相关题目

⭐有效的括号

⭐逆波兰表达式求值

⭐栈的压入、弹出序列

💯队列相关题目

⭐用队列实现栈

⭐用栈实现队列

💯总结


💯前言

在技术面试中,栈和队列是数据结构领域的重要考点,经常会出现各种相关的题目来考查应聘者对这两种数据结构的理解和掌握程度。以下是一些栈和队列面试中常考的题目及解析

 

💯栈相关题目

⭐有效的括号

题目链接👉【力扣】

题目描述:给定一个只包含括号 '('')''{''}''['']' 的字符串,判断字符串中的括号是否有效。有效字符串需满足:

  1. 左括号必须用相同类型的右括号闭合。
  2. 左括号必须以正确的顺序闭合。

示例 1
输入:"()"
输出:true

示例 2
输入:"()[]{}"
输出:true

示例 3
输入:"(]"
输出:false

  

解题思路

我们可以使用来解决这个问题。遍历字符串中的每个字符,如果是左括号,就将其压入栈中。如果是右括号,就检查栈顶元素是否是对应的左括号,如果是,则弹出栈顶元素;如果不是,或者栈为空,则说明括号无效。最后,如果栈为空,则说明所有括号都匹配成功,字符串中的括号是有效的;否则,括号无效。

代码实现(C++)

class Solution {
public:
    bool isValid(string s) {
        stack<char> st;
        for (char c : s) {
            if (c == '(' || c == '[' || c == '{') {
                st.push(c);
            } else {
                if (st.empty()) return false;
                char top = st.top();
                if ((c == ')' && top == '(') || (c == ']' && top == '[') || (c == '}' && top == '{')) {
                    st.pop();
                } else {
                    return false;
                }
            }
        }
        return st.empty();
    }
};

⭐逆波兰表达式求值

题目链接👉【力扣】

题目描述:逆波兰表达式是一种后缀表达式,它将运算符放在操作数之后。例如,表达式 3 4 + 5 * 对应的中缀表达式是 (3 + 4) * 5。给定一个逆波兰表达式,求其求值结果。

示例
输入:["2", "1", "+", "3", "*"]
输出:9
解释:该逆波兰表达式的计算过程为:(2 + 1) * 3 = 9

解题思路

使用来解决这个问题。遍历逆波兰表达式中的每个元素,如果是数字,就将其转换为整数并压入栈中。如果是运算符,就从栈中弹出两个操作数,进行相应的运算,然后将结果压回栈中。最后,栈中剩下的唯一元素就是表达式的求值结果。

代码实现(C++)

class Solution {
public:
    int evalRPN(vector<string>& tokens) {
        stack<int> st;
        for (string token : tokens) {
            if (token == "+" || token == "-" || token == "*" || token == "/") {
                int num2 = st.top(); st.pop();
                int num1 = st.top(); st.pop();
                if (token == "+") st.push(num1 + num2);
                else if (token == "-") st.push(num1 - num2);
                else if (token == "*") st.push(num1 * num2);
                else st.push(num1 / num2);
            } else {
                st.push(stoi(token));
            }
        }
        return st.top();
    }
};

⭐栈的压入、弹出序列

题目描述:输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如,序列 1,2,3,4,5 是某栈的压栈序列,序列 4,5,3,2,1 是该压栈序列对应的一个弹出序列,但 4,3,5,1,2 就不可能是该压栈序列的弹出序列。

解题思路

使用一个辅助栈来模拟栈的压入和弹出操作。遍历压入序列,将元素依次压入辅助栈。同时,遍历弹出序列,如果辅助栈的栈顶元素与弹出序列的当前元素相等,就将辅助栈的栈顶元素弹出,并继续比较下一个弹出元素。如果最后辅助栈为空,则说明弹出序列是可能的;否则,弹出序列是不可能的。

代码实现(C++)

class Solution {
public:
    bool validateStackSequences(vector<int>& pushed, vector<int>& popped) {
        stack<int> st;
        int i = 0, j = 0;
        while (i < pushed.size()) {
            st.push(pushed[i++]);
            while (!st.empty() && st.top() == popped[j]) {
                st.pop();
                j++;
            }
        }
        return st.empty();
    }
};

💯队列相关题目

⭐用队列实现栈

题目链接👉

题目描述:请你仅使用两个队列实现一个后入先出(LIFO)的栈,并支持普通栈的全部四种操作pushtoppop 和 empty

实现思路

  1. 入栈操作(push:将元素入队到一个非空队列中(如果两个队列都为空,任选一个即可)。
  2. 出栈操作(pop:将除了最后一个元素之外的所有元素从当前非空队列依次出队并入队到另一个空队列中,然后将最后一个元素出队,这个元素就是栈顶元素。
  3. 获取栈顶元素(top:与出栈操作类似,将除了最后一个元素之外的所有元素从当前非空队列依次出队并入队到另一个空队列中,然后记录最后一个元素的值(但不出队),最后再将这个元素入队回原来的队列。
  4. 判断栈是否为空(empty:检查两个队列是否都为空。

代码实现(C++)

class MyStack {
private:
    queue<int> q1;
    queue<int> q2;

public:
    void push(int x) {
        if (q1.empty()) {
            q2.push(x);
        } else {
            q1.push(x);
        }
    }

    int pop() {
        int element;
        if (q1.empty()) {
            while (q2.size() > 1) {
                q1.push(q2.front());
                q2.pop();
            }
            element = q2.front();
            q2.pop();
        } else {
            while (q1.size() > 1) {
                q2.push(q1.front());
                q1.pop();
            }
            element = q1.front();
            q1.pop();
        }
        return element;
    }

    int top() {
        int element;
        if (q1.empty()) {
            while (q2.size() > 1) {
                q1.push(q2.front());
                q2.pop();
            }
            element = q2.front();
            q1.push(element);
            q2.pop();
        } else {
            while (q1.size() > 1) {
                q2.push(q1.front());
                q1.pop();
            }
            element = q1.front();
            q2.push(element);
            q1.pop();
        }
        return element;
    }

    bool empty() {
        return q1.empty() && q2.empty();
    }
};

⭐用栈实现队列

(一)思路分析

要用栈实现队列,同样需要利用两个栈。一个栈用于入队操作(称为输入栈),另一个栈用于出队操作(称为输出栈)。当进行入队操作时,将元素直接压入输入栈。当进行出队操作时,如果输出栈为空,将输入栈中的所有元素依次弹出并压入输出栈,然后从输出栈中弹出顶部元素,这个元素就是队头元素。

(二)代码实现

  1. 结构体定义
    typedef struct {
        Stack *inputStack;
        Stack *outputStack;
    } QueueUsingStacks;

这里定义了一个结构体,包含两个指向栈的指针,用于实现队列的功能。


2. 初始化队列

    void initQueueUsingStacks(QueueUsingStacks *queue) {
        queue->inputStack = (Stack *)malloc(sizeof(Stack));
        queue->outputStack = (Stack *)malloc(sizeof(Stack));
        initStack(queue->inputStack);
        initStack(queue->outputStack);
    }

初始化时,为两个栈分配内存空间,并分别初始化它们。


3. 入队操作

    void enqueueUsingStacks(QueueUsingStacks *queue, int element) {
        pushStack(queue->inputStack, element);
    }

入队时,将元素压入输入栈


4. 出队操作

    int dequeueUsingStacks(QueueUsingStacks *queue) {
        int element;
        if (isEmptyStack(queue->outputStack)) {
            while (!isEmptyStack(queue->inputStack)) {
                pushStack(queue->outputStack, popStack(queue->inputStack));
            }
        }
        if (!isEmptyStack(queue->outputStack)) {
            element = popStack(queue->outputStack);
            return element;
        } else {
            printf("队列为空,无法出队\n");
            return -1;
        }
    }

出队时,如果输出栈为空,将输入栈中的元素转移到输出栈中,然后从输出栈中弹出顶部元素。

 

💯总结

 

通过以上对栈和队列面试常见题目的分析与解答,希望能帮助你更好地理解和掌握这两种数据结构在面试中的应用,提高你在数据结构方面的解题能力和应对面试的信心。在实际准备面试过程中,建议你多做练习,深入理解每种数据结构的特点和操作方法,以便能够灵活运用它们解决各种问题。如果你还有其他问题或需要进一步的解释,欢迎随时提问。


💝💝💝感谢你看到最后,点个赞再走吧!💝💝💝 

以下是一个投票,欢迎你参与,让我们一起了解大家对栈和队列的认知和兴趣程度: 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

双十一好物清单有哪些值得入手?双11好物种草宝藏清单分享

随着双十一购物狂欢节的临近&#xff0c;各种优惠和折扣让人目不暇接&#xff0c;在这个充满诱惑的时刻&#xff0c;如何挑选出真正值得入手的好物&#xff0c;成为了许多消费者的难题&#xff0c;为此&#xff0c;我精心整理了一份双十一好物清单&#xff0c;为大家提供一些实…

Linux 网络配置 (深入理解)

前言 前期我比较迷惑Ubuntu 的网络配置。 我接触比较多的 Linux 发行版都是 Ubuntu &#xff0c;我按照网上的一些教程配置网络发现&#xff0c;没有相关网络配置文件夹。然后我发现不是我的问题而是不同版本的配置方式和工具是不一样的。然后有些配置已经弃用了。 常见的网络…

计算机网络--TCP、UDP抓包分析实验

计算机网络实验 目录 实验目的 实验环境 实验原理 1、UDP协议 2、TCP协议 实验具体步骤 实验目的 1、掌握使用wireshark工具对UDP协议进行抓包分析的方法&#xff0c;掌握UDP协议的报文格式&#xff0c;掌握UDP协议校验和的计算方法&#xff0c;理解UDP协议的优缺点&am…

手把手教你使用YOLOv11训练自己数据集(含环境搭建 、数据集查找、模型训练)

一、前言 本文内含YOLOv11网络结构图 训练教程 推理教程 数据集获取等有关YOLOv11的内容&#xff01; 官方代码地址&#xff1a;https://github.com/ultralytics/ultralytics/tree/main/ultralytics/cfg/models/11 二、整体网络结构图 三、环境搭建 项目环境如下&#xf…

verilog实现FIR滤波系数生成(阶数,FIR滤波器类型及窗函数可调)

在以往采用 FPGA 实现的 FIR 滤波功能&#xff0c;滤波器系数是通过 matlab 计算生成&#xff0c;然后作为固定参数导入到 verilog 程序中&#xff0c;这尽管简单&#xff0c;但灵活性不足。在某些需求下&#xff08;例如捕获任意给定台站信号&#xff09;需要随时修改滤波器的…

FPGA学习(1)-mux2,2选1多路器

目录 1 开发板配套资料 1.1学习网址和资料网址 2.创建工程文件 2.1创建过程 2.2写程序及仿真测试 2.2.1 写程序生成电路 2.2.2仿真 2.2.3 生成执行文件并烧录 3.实验现象 买的小梅哥店铺的开发板&#xff1a;xc7z020clg400 看的小梅哥的视频&#xff1a;03C _基于ZYN…

VUE 开发——AJAX学习(三)

一、async函数和await async和await关键字让我们可以用一种更简洁的方式写出基于Promise的异步行为&#xff0c;而无需刻意地链式调用Promise async写在函数声明的前面&#xff1b;在async函数内&#xff0c;使用await关键字&#xff0c;获取Promise对象“成功状态”结果值 &…

身份证号、定位信息等个人信息敏感性判定解析

关于身份证号号码以及精确定位信息是否是敏感个人信息的疑问一直以来不少合规安全从业者有疑惑&#xff0c;本文来自于《数安标准强基助力计划 》作者为指南和标准的起草者&#xff0c;其观点具有一定的权威性&#xff0c;一下为内容摘要&#xff0c;以供大家学习和解惑&#x…

Qt多线程操作sqlite数据库

问题 就是为了多线程操作sqlite数据库,为什么,因为数据库是耗时的操作,一条数据的插入,差不多200ms,如果是数据插入多了,界面会有明显的卡顿,因此必须,多线程操作数据库。 问题是这样的: 插入数据之后,接着更新界面;然而,插入数据是比较耗时的操作,尤其插入数据…

图解C#高级教程(一):委托

什么是委托 可以认为委托是持有一个或多个方法的对象。但它与对象不同&#xff0c;因为委托可以被执行。当执行委托时&#xff0c;委托会执行它所“持有”的方法。先看一个完整的使用示例。 // See https://aka.ms/new-console-template for more informationdelegate void M…

【Git原理与使用】Git初识基本操作

Git初识&&基本操作 1.初识Git2.Git安装3.Git基本操作3.1创建Git本地仓库3.2配置Git3.3认识工作区、暂存区、版本库3.4添加文件3.5修改文件3.6版本回退3.7撤销修改3.8删除文件 点赞&#x1f44d;&#x1f44d;收藏&#x1f31f;&#x1f31f;关注&#x1f496;&#x1f…

Vscode超好看的渐变主题插件

样式效果&#xff1a; 插件使用方法&#xff1a; 然后重启&#xff0c;之后会显示vccode损坏&#xff0c;不用理会&#xff0c;因为这个插件是更改了应用内部代码&#xff0c;直接不再显示即可。

基于nodejs+vue的游戏陪玩系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码 精品专栏&#xff1a;Java精选实战项目…

银河麒麟服务器:更新软件源

银河麒麟服务器&#xff1a;更新软件源 1、使用场景2、操作步骤3、注意事项 &#x1f490;The Begin&#x1f490;点点关注&#xff0c;收藏不迷路&#x1f490; 1、使用场景 当需要安装最新软件或修改软件源配置后&#xff0c;需更新软件源以获取最新软件包信息。 2、操作步…

京东PMO段敬受邀为第四届中国项目经理大会演讲嘉宾

全国项目经理专业人士年度盛会 京东集团技术委员会PMO组研发项目经理段敬女士受邀为PMO评论主办的全国项目经理专业人士年度盛会——2024第四届中国项目经理大会演讲嘉宾&#xff0c;演讲议题为“项目经理如何组织高效的项目会议”。大会将于10月26-27日在北京举办&#xff0c;…

亚信安全天穹5分钟勒索体检 免费试用今起上线

对于勒索攻击的认知 你是否还停留在“2.0时代”&#xff1f; 勒索攻击无疑是企业面临的最大威胁&#xff0c;2024年上半年&#xff0c;勒索组织数量同步增长超过50%&#xff0c;勒索攻击数量也持续攀升&#xff0c;平均勒索赎金突破520万美元。 当前&#xff0c;勒索攻击治理…

java学习-idea编辑器基础使用设置

首先打开电脑中的idea编辑器&#xff0c;点击头部&#xff1a;File按钮 → Settings… 打开设置界面&#xff1b; 设置idea的主题 设置idea代码注释的字体颜色 设置idea编辑器的字体和字体大小 设置idea通过提示回车自动导入包 设置idea输入忽略大小写进行提示

【Hadoop】【vim编辑器】【~/.bashrc 文件】如何编辑

1. 进入 vim 编辑器 在终端中输入以下命令&#xff1a; vim ~/.bashrc 2. 进入插入模式 打开文件后&#xff0c;你将处于普通模式。在普通模式下&#xff0c;你不能直接编辑文本。 要进入插入模式&#xff0c;请按下 i 键。这时&#xff0c;你应该会看到屏幕底部出现 -- 插…

fish-speech语音大模型本地部署

文章目录 fish-speech模型下载编译部署 小结 fish-speech模型 先说下fish-speech模型吧&#xff0c;可以先看下官网。如下&#xff1a; 这就是一个模型&#xff0c;可以根据一个样例声音&#xff0c;构建出自己需要的声音。其实&#xff0c;这个还是有很多用途的&#xff1b;…

多模态人像编辑:PortraitGen将2D肖像视频提升到4D 高斯场

这篇文章《Portrait Video Editing Empowered by Multimodal Generative Priors》&#xff0c;作者是来自中国科学技术大学。文章介绍了一种名为PortraitGen的肖像视频编辑方法&#xff0c;它使用多模态生成先验来实现一致性和富有表现力的风格化编辑。 文章地址&#xff1a;P…