| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 数据结构与算法 -> 【从零开始】不可分割的栈和队列 -> 正文阅读 |
|
[数据结构与算法]【从零开始】不可分割的栈和队列 |
栈和队列是非常经典的数据结构,基本上是随处可见了。 为什么把这两者放在一起说呢? 因为对于C++的STL来说,这两者都是容器适配器,默认都是以deque为底层实现的。 我们都知道栈是【先进后出】,队列是【先进先出】 使用双端队列deque并且进行一些加工就可以实现栈和队列的效果,这也是STL中栈和队列的实现方式。 具体细节可以看这篇笔记: 栈和队列互相实现有一类题目是栈和队列的相互实现,比如用栈实现队列或者用队列实现栈。 虽然没有太明显的实际意义,但是对于理解栈和队列的结构至关重要。 用栈实现队列用栈实现队列就要思考队列的特点,那些是栈需要调整的。 入队列和入栈一致,不需要修改。 而出队和出栈恰好相反,所以我们需要两个栈来对栈内的数据进行处理。 比如想实现1,2,3入队列然后1出队。 这时候要把1号栈中的[1,2,3]全部出栈,再放入2号栈变为[3,2,1],这时出栈的就是队头。 如果1出队后,变为[2,3],此时[4,5]又入栈了怎么办呢? 这就要看2号栈里是否还有数据,如果有那么还是存在队头的,不需要把新数据倒入。
需要注意: 在stack中的pop()会直接把栈顶元素删除,而不会返回值。所以想要实现题目中的pop()需要先记录再pop() peek()这个地方直接复用了写好的pop()函数,因为需要队列的头元素,所以复用更高效。 用队列实现栈队列实现栈就要考虑栈的特点和队列有什么差异。 队列入队和栈入栈还是一致的,不需要改动。 出队因为是队头,而栈需要出的元素在队尾,所以需要调整。 这里我们不使用新的空间,直接把前面的元素搬到队尾的后面,再次入队。 比如[1,2,3,4],弹出栈头应该是4,我们把前面的元素重新入队,得到[4,1,2,3],再弹出4即可。
同理,top就可以直接复用写好的pop()。 符号匹配栈的一大作用就是符号的匹配,我们来看两道这样的题目 括号匹配在判断括号匹配时用栈会方便很多。 给定一个字符串,判断括号是否有效。 "()[]{}“这样就是有效的组合,而”[)"则不是有效的组合。 需要特别注意,"([)]"这种类型即使看上去数量是对的,也不符合正确的括号闭合。 核心: 我们的核心做法就是使用栈进行匹配。 遇到比如"(“,我们直接存入栈”)“,因为如果结果正确那么后面也会有”)",这样就可以相互抵消。 抵消过程就是当前的元素和栈顶比较,如果相等就出栈即可,一层一层解开嵌套。 如果抵消失败说明不符合要求。 最后如何判断是否成功呢?只要循环结束后栈空就说明成功。 我们在循环内也有栈空的情况,也就是没有遇到左括号,这个时候返回false。 而整个循环外的栈空就说明已经全部配对了,所以是true。
逆波兰表达式逆波兰式是计算机中一种重要的算术表达方式。 比如(2+1)*3,我们需要使用括号来表示优先级。 而改为逆波兰式就是2 1 + 3 * 解释为2和1用+计算,然后结果再与3用*计算,这样不需要括号也可以区分优先级。 还有一点小知识,逆波兰式其实是二叉树的后序遍历。 我们以运算符为父节点,数字为子节点就可以画出这棵树。 给出逆波兰表达式,求出运算结果。 我们依旧使用栈,在遇到数字时入栈,遇到符号就把前两个数提出来做计算,再将新值入栈。
单调栈单调栈是以栈为基础的一种数据结构,但是做了一些属性的修改,保证栈内的元素具有单调性。 基本逻辑先来看一道题 给定一个nums1和nums2,根据nums1找到nums2对应的元素,然后在nums2中寻找右侧第一个比该元素大的数。 比如nums1=[4,1],nums2=[1,3,4,2] 对于4来说,在nums2中4的右侧没有比它大的,所以返回-1 对于1来说,在nums2中1的右侧第一个比它大的是3 最后返回[-1,3] 这就是整道题的逻辑 单调栈的遍历方法在这之前,我们先简化上一道题的场景。 对于nums1和nums2,我们去掉nums1,留下nums2。 只找nums2对应元素的右侧第一个比它大的元素。 比如[1,3,4,2] 返回结果应该是[3,4,-1,-1] 从这里开始我们构建单调栈。 核心: 如果我们将[1,3,4,2]入栈遍历,应该如何进行呢? 因为我们要找的是右侧第一个比它大的元素,要看右边,所以最好的入栈方式是倒着入栈。 上例中就应该按照2,4,3,1的顺序来。 判断条件也比较清楚:如果当前这个数比已经入栈的元素(栈顶)大(或者等于),那说明已经入栈的元素不是它右侧第一个大的元素,直接把入栈元素弹出即可。 最后这个位置的答案就是栈顶元素,没有即为-1 按2,4,3,1走一遍。 2先遍历,因为栈为空,所以表示没有2右侧大于它的元素,返回-1,2入栈 然后遍历到4,因为4>2,所以2不是目标值,故弹出2。然后返回-1,4入栈 然后到3,因为3<4,所以4是目标值,结果为栈顶4,然后3入栈 最后是1,因为1<3,所以3是目标值,结果为栈顶3,然后1入栈 最后返回结果数组即可。
再回到我们的题目中,因为有nums1和nums2,我们可以用一个哈希表存放nums2和对应的下一个更大元素,然后遍历nums1将对应的哈希表中的数值取出即可。
每日温度再看一道变体的题目,大同小异。 给定一个数组,返回的是再走几个数就可以得到比它大的下一个元素。 与上题的区别就是,上题返回的是比它大的下一个数,而该题返回的是距离。 返回距离也很简单,我们只需要操作下标就可以了。把原来栈内的元素换成下标。
逻辑和上题没有任何区别 升级版下一个更大元素下一个更大元素还有一道升级版的题目: 给定一个数组,比如[1,2,1],按照我们的正常求法求下一个更大元素,返回的是[2,-1,-1] 而本题让数组循环,1并不是结尾,而是继续回到起始位置重新循环。 这时返回的就是[2,-1,2] 很明显,我们只要把给定的数组double,变成[1,2,1,1,2,1]就可以正确返回前三个元素的下一个更大元素了。 而对于for循环来说,因为控制的是下标,所以并不需要真正的去扩展数组。
这里的精髓就是将for循环的长度变为两倍,然后使用取模的方法使结果只存在长度为n的数组中。 单调队列单调队列,和单调栈性质一样,使队列维持一个具有单调性的状态。 想要让队列保持单调性,就要在入队和出队的时候做一些处理。 这里我们直接来看一道经典题目,体会自己实现单调队列的方法。 给定一个数组和窗口大小,返回这个窗口内的最大值,然后移动窗口。 输入:nums = [1,3,-1,-3,5,3,6,7], k = 3 [1 3 -1] -3 5 3 6 7 3 单调队列的核心这个窗口的移动很像队列的出入队操作。但是我们想让队列直接帮我们求出最大值。 所以我们实现一个单调队列,以deuqe为基础(因为可以在队头和队尾进行插入删除操作,很灵活) 单调队列有这样几个重点:
举个例子:[1,3,-1,-3,5,3,6,7] 1入队,3入队时发现比1大,所以1出队。 -1入队,这时候队列为[3,-1] 窗口移动,因为需要移出去的1和队列的3不匹配,所以队列不需要动。 -3入队,队列为[3,-1,-3] 窗口移动,这时候需要移出去的是3,队头也是3,所以pop。 5入队,发现比-3大,删除-3,又比-1大。,删除-1 这时候的队列为[5] …后面以此类推。 所以单调队列的实现我们就可以写出来了: deque用到的操作有:push_front(),push_back(),pop_front(),pop_back()
有了单调队列,我们就可以根据窗口移动的方式,往队列里放元素,让单调队列给我们最大值。
其他还有一些与优先级队列有关的题目,我们在总结完二叉树之后继续探究。 |
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 | -2025/1/4 16:53:52- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |