数据结构-栈和队列-2

简介: 单调栈是栈的一种特例,在一般情况下使用较少。单调栈分为单调递增栈和单调递减栈。单调递增栈是栈中元素从栈底到栈顶是递增的。单调递减栈是栈中元素从栈底到栈顶是递减的。

四、单调栈


  • 单调栈是栈的一种特例,在一般情况下使用较少。
  • 单调栈分为单调递增栈和单调递减栈。
  • 单调递增栈是栈中元素从栈底到栈顶是递增的。


  • 单调递减栈是栈中元素从栈底到栈顶是递减的。
  • 应用:求解下一个大于 x 元素或者是小于 x 元素的位置。


给一个数组,返回一个大小相同的数组,返回的数组的第 i 个位置的值应当是,对于原数组中的第 i 个元素,至少往右走多少步,才能遇到一个比自己大的元素(如果之后没有比自己大的元素,或者已经是最后一个元素,则在返回数组的对应位置放上 -1 )。9d63b258fe514cc4b532241e2b608e16.png


1. 伪代码

stack<int> st;
for (遍历这个数组){
  if (栈空 || 栈顶元素大于等于当前比较元素){
    入栈;
  }
  else{
    while (栈不为空 && 栈顶元素小于当前元素){
      栈顶元素出栈;
      更新结果;
    }
    当前数据入栈;
  }
}



2. 单调栈的应用


  • 左边区间第一个比它小的数,第一个比它大的数。
  • 确定这个元素是否是区间最值。
  • 右边区间第一个大于它的值。
  • 到 右边区间第一个大于它的值的距离。
  • 确定以该元素为最值的最长区间。



五、单调队列


  • 单调队列是队列的一种特例,在一般情况下使用较少。
  • 单调队列中的元素始终保持着单增或者单减的特性。


  • 单调队列不只是做到了有序,还可以在每次加入或者删除元素时都保持序列里的元素有序,即队首元素始终是最小值或者最大值。
  • 单调队列分为单调递增队列和单调递减队列两种。


  • 单调递增队列:保证队列头元素一定是当前队列的最小值,用于维护区间的最小值。
  • 单调递减队列:保证队列头元素一定是当前队列的最大值,用于维护区间的最大值。
  • 实现单调队列,主要分为三个部分:


去尾操作 :队尾元素出队列。当队列有新元素待入队,需要从队尾开始,删除影响队列单调性的元素,维护队列的单调性。(删除一个队尾元素后,就重新判断新的队尾元素)。


删头操作 :队头元素出队列。判断队头元素是否在待求解的区间之内,如果不在,就将其删除。(这个很好理解呀,因为单调队列的队头元素就是待求解区间的极值)。


取解操作 :经过上面两个操作,取出 队列的头元素 ,就是 当前区间的极值 。

单调队列与普通队列有些不同,因为右端既可以插入又可以删除,因此在代码中通常用一份数组和 两个指针来实现,而不是使用 STL 中的 queue。如果一定要使用 STL ,那么则可以使用双端队列(即两端都可以插入和删除),即deque 。

————————————————


1. 单调队列的应用


  • 可以查询区间最值(不能维护区间k大,因为队列中很有可能没有k个元素)。
  • 优化DP,单调队列一般是用于优化动态规划方面问题的一种特殊数据结构,且多数情况是与定长连续子区间问题相关联。


2. 与单调栈的异同

2.1 相同点


  • 单调队列和单调栈的“头部”都是最先添加的元素,“尾部”都是最后添加的元素。
  • 递增和递减的判断依据相同:从栈底(队尾)到栈顶(队首),元素大小的变化情况。所以队列和栈是相反的。
  • 两者维护的时间复杂度都是 O(n),每个元素都只操作一次。


2.2 不同点



单调队列可以从队列头弹出元素,可以方便地根据入队的时间顺序(访问的顺序)删除元素。


单调队列和单调栈维护的区间不同。当访问到第i个元素时,单调栈维护的区间为 [0,i),而单调队列维护的区间为 (lastpop,i)


单调栈无法获取 [0,i) 的区间最大值/最小值。因为单调队列可以访问“头部”和“尾部”,而单调栈只能访问栈顶(也就是“尾部”)。




相关文章
|
2天前
|
存储 C语言
数据结构基础详解(C语言): 栈与队列的详解附完整代码
栈是一种仅允许在一端进行插入和删除操作的线性表,常用于解决括号匹配、函数调用等问题。栈分为顺序栈和链栈,顺序栈使用数组存储,链栈基于单链表实现。栈的主要操作包括初始化、销毁、入栈、出栈等。栈的应用广泛,如表达式求值、递归等场景。栈的顺序存储结构由数组和栈顶指针构成,链栈则基于单链表的头插法实现。
|
4天前
|
Java
【数据结构】栈和队列的深度探索,从实现到应用详解
本文介绍了栈和队列这两种数据结构。栈是一种后进先出(LIFO)的数据结构,元素只能从栈顶进行插入和删除。栈的基本操作包括压栈、出栈、获取栈顶元素、判断是否为空及获取栈的大小。栈可以通过数组或链表实现,并可用于将递归转化为循环。队列则是一种先进先出(FIFO)的数据结构,元素只能从队尾插入,从队首移除。队列的基本操作包括入队、出队、获取队首元素、判断是否为空及获取队列大小。队列可通过双向链表或数组实现。此外,双端队列(Deque)支持两端插入和删除元素,提供了更丰富的操作。
10 0
【数据结构】栈和队列的深度探索,从实现到应用详解
|
8天前
|
Linux C++ Windows
栈对象返回的问题 RVO / NRVO
具名返回值优化((Name)Return Value Optimization,(N)RVO)是一种优化机制,在函数返回对象时,通过减少临时对象的构造、复制构造及析构调用次数来降低开销。在C++中,通过直接在返回位置构造对象并利用隐藏参数传递地址,可避免不必要的复制操作。然而,Windows和Linux上的RVO与NRVO实现有所不同,且接收栈对象的方式也会影响优化效果。
|
23天前
|
存储 安全 编译器
缓冲区溢出之栈溢出(Stack Overflow
【8月更文挑战第18天】
46 3
|
24天前
|
测试技术
【初阶数据结构篇】栈的实现(附源码)
在每一个方法的第一排都使用assert宏来判断ps是否为空(避免使用时传入空指针,后续解引用都会报错)。
|
10天前
crash —— 获取内核地址布局、页大小、以及栈布局
crash —— 获取内核地址布局、页大小、以及栈布局
|
10天前
|
存储 程序员 C语言
堆和栈之间有什么区别
【9月更文挑战第1天】堆和栈之间有什么区别
77 0
|
19天前
|
机器学习/深度学习 消息中间件 缓存
栈与队列的实现
栈与队列的实现
35 0
|
24天前
|
测试技术
【初阶数据结构篇】队列的实现(赋源码)
首先队列和栈一样,不能进行遍历和随机访问,必须将队头出数据才能访问下一个,这样遍历求个数是不规范的。
|
29天前
|
算法 C语言 C++
【practise】栈的压入和弹出序列
【practise】栈的压入和弹出序列