【栈和队列高频考点题】(一)

简介: 【栈和队列高频考点题】(一)

1 与栈有关的考题

1.1 最小栈

题目描述:

8f751949183d4e6f850b29bb13795ff7.png 解题思路:

要想在O(1)时间内获得栈内最小元素只用一个栈肯定是行不通的,所以我们不妨再开一个栈来记录当前栈里面每个元素的最小值,这样就要多开出O(N)的空间,可以优化空间的方法是并不是所有的元素都要入最小栈,而是当前元素比最小栈栈顶的数据大便不必再入数据了,但是等于时一定要入进去,当pop掉元素时只需要比较普通栈与最小栈栈顶元素是否相等,若相等,则最小栈栈顶数据也得pop。

参考代码:

class MinStack {
public:
    MinStack() {}
    void push(int val) {
        _st.push(val);
        if(_minSt.empty() || _minSt.top()>=val)
            _minSt.push(val);
    }
    void pop() {
         if(_minSt.top()==_st.top())
            _minSt.pop();
         _st.pop();
    }
    int top() {
        return _st.top();
    }
    int getMin() {
        return _minSt.top();
    }
    stack<int> _st;
    stack<int> _minSt;//记录最小的数据
};

运行结果:

d8ee5956cf6546f18e69f5986446c072.png

题后反思:

万一数据像这种22222222,貌似我们优化空间的方法起不到作用了,可以采取的优化措施是我们存储数据时可以存一个pair,里面额外记录一下变量的个数,当有重复数据时便不再入数据了,而是++里面的计数器,pop时就--里面的计数器,大家有兴趣可以自己实现下,这里我就不再实现了。

1.2 栈的弹出压入序列

题目描述:

095bef7dc61d48df959b3380ec88709b.png

解题思路:

这道题的解决方法有很多种,这里分享一种比较容易理解的思路:另外搞一个栈,将栈的压入序列不断的push到这个栈中,当该栈不为空并且栈的弹出序列与该栈栈顶数据相等时就pop掉该栈栈顶数据,然后不断迭代走,最后返回值就是该栈是否为空或者是否走到了弹出序列的尾。

参考代码:

class Solution {
public:
    bool IsPopOrder(vector<int> pushV,vector<int> popV) {
        stack<int> v;
        int pushi=0,popi=0;
        while(pushi<pushV.size())
        {
            v.push(pushV[pushi++]);
            while(!v.empty() && v.top()==popV[popi])
            {
                v.pop();
                popi++;
            }
        }
        return v.empty();
    }
};

运行结果:

8d1a71fda5aa4b4683aa3f767bb9cb5e.png

1.3 逆波兰表达式求值

题目描述:

337281658a084cd8b0b13551e47b5579.png

解题思路:

由于逆波兰表达式就是后缀表达式,运算符的优先级已经确定了,所以我们只需要模拟下运算的过程即可,注意先取的数据是右,后取的数据为左。

参考代码:

class Solution {
public:
    int evalRPN(vector<string>& tokens) {
        stack<int> s;
        for(auto& e:tokens)
        {
            if(e=="+" || e=="-" || e=="*" || e=="/")
            {
                int right=s.top();
                s.pop();
                int left=s.top();
                s.pop();
                switch(e[0])
                {
                    case '+':
                        s.push(left+right);
                        break;
                    case '-':
                        s.push(left-right);
                        break;
                    case '*':
                        s.push(left*right);
                        break;
                    case '/':
                        s.push(left/right);
                        break;
                }
            }
            else
            {
                s.push(stoi(e));
            }
        }
        return s.top();
    }
};

运行结果:

aa5bcea21fa74a0c8d6a703ed40d14b9.png

题后反思:

假如给的不是后缀表达式,而是中缀表达式,如何将中缀表达式转化为后缀表达式呢?

给定中缀表达式【1+2*(3-4)+5/6】,我们如何将其转化为后缀表达式?

基本规则是这样的:当遇到操作数的时候我们直接输出;遇到操作符当栈为空就直接入栈,栈若不为空就与栈顶数据相比,若优先级大于栈顶就将该操作符入栈(由于后面操作符优先级可能更大,所以先让其入栈),若优先级小于或者等于栈顶,就将栈顶数据pop掉然后输出,直到栈为空或者遇到优先级更小的操作符,然后继续迭代直到访问到所有元素。

但是这个规则只适用于没有()的转换中,例如【1+2*3-4】可以转化成【1 2 3 * + 4 -】

但是像上面的那个栗子【1+2*(3-4)+5/6】貌似就不适用了,处理方法有两种:

第一种是不入()到栈中,当发现出现了()时就认为()中的运算符优先级非常大,就让其入栈,其他规则与基本规则类似。

第二种是将()入栈,当出现(时就认为(的优先级非常低,(目的是让括号中运算符入栈),当出现)时同样认为)的优先级非常低,(目的是让括号中运算符出栈输出)

无论使用哪种规则,我们都能够很轻易的将上述中缀表达式转化为:

【1 2 3 4 - * + 5 6 / +】


目录
相关文章
|
3天前
|
算法 安全 测试技术
golang 栈数据结构的实现和应用
本文详细介绍了“栈”这一数据结构的特点,并用Golang实现栈。栈是一种FILO(First In Last Out,即先进后出或后进先出)的数据结构。文章展示了如何用slice和链表来实现栈,并通过golang benchmark测试了二者的性能差异。此外,还提供了几个使用栈结构解决的实际算法问题示例,如有效的括号匹配等。
golang 栈数据结构的实现和应用
|
3天前
|
前端开发
07_用队列实现栈
07_用队列实现栈
06_用栈来求解汉诺塔问题
06_用栈来求解汉诺塔问题
05_用一个栈实现另一个栈的排序
05_用一个栈实现另一个栈的排序
03_如何仅用递归函数和栈操作逆序一个栈
03_如何仅用递归函数和栈操作逆序一个栈
|
3天前
|
测试技术
02_由两个栈组成的队列
02_由两个栈组成的队列
01_设计一个有getMin功能的栈
01_设计一个有getMin功能的栈
|
7天前
|
存储
|
22天前
|
存储 人工智能 C语言
数据结构基础详解(C语言): 栈的括号匹配(实战)与栈的表达式求值&&特殊矩阵的压缩存储
本文首先介绍了栈的应用之一——括号匹配,利用栈的特性实现左右括号的匹配检测。接着详细描述了南京理工大学的一道编程题,要求判断输入字符串中的括号是否正确匹配,并给出了完整的代码示例。此外,还探讨了栈在表达式求值中的应用,包括中缀、后缀和前缀表达式的转换与计算方法。最后,文章介绍了矩阵的压缩存储技术,涵盖对称矩阵、三角矩阵及稀疏矩阵的不同压缩存储策略,提高存储效率。
|
24天前
|
存储 C语言
数据结构基础详解(C语言): 栈与队列的详解附完整代码
栈是一种仅允许在一端进行插入和删除操作的线性表,常用于解决括号匹配、函数调用等问题。栈分为顺序栈和链栈,顺序栈使用数组存储,链栈基于单链表实现。栈的主要操作包括初始化、销毁、入栈、出栈等。栈的应用广泛,如表达式求值、递归等场景。栈的顺序存储结构由数组和栈顶指针构成,链栈则基于单链表的头插法实现。
147 3