数据结构·单调栈和单调队列

思想

  • 本质都是讲原本 O ( n 2 ) O(n^2) O(n2)的算法优化为 O ( n ) O(n) O(n),都是利用使用一个容器维护某段区间,节省一段for循环,同时这段区间维持单调性
  • 出发点:删除容器内的冗余元素。

单调栈

例题:830. 单调栈

应用:左边第1个小于当前数的数

  • 暴力做法是两个for循环,第一层循环表示当前数,第二层循环用于搜索第一个小于当前数的数。
  • 使用容器存储第二个for循环中遍历的数,意味着我们可以去掉一重for循环。
    • 方便查询目标数
    • 因此我们想到可以优化容器内的数据存储,删除冗余的数据
  • 删除容器中的冗余元素 :我们发现如果i<j,且ai>aj,那么ai一定是当前容器中的冗余元素,我们就可以删去它。此时容器末位的元素就是我们需要的第1个小于当前数的数。(如果容器为空,则返回-1)
  • 然后按照这个规则,容器内的元素就变成单调递增了,然后我们用栈方便我们的实现。
cpp 复制代码
int n, a;
stack<int>st;
void solve() {
	cin >> n;
	for (int i = 1; i <= n; i++) {
		cin >> a;
		while (!st.empty() && st.top() >= a) {
			st.pop();
		}
		if (st.empty())cout << -1 << " ";
		else {
			cout << st.top() << " ";
		}
		st.push(a);
	}
}

单调队列

例题:154. 滑动窗口

应用:滑动窗口最大值或者最小值

  • 暴力方法遍历每一个窗口和窗口内元素
  • 尝试使用一个容器维护当前窗口:
    • 这个容器需要实现窗口移动时容器大小减少以保持一致
    • 方便查询最大值或者最小值。
    • 因此我们想到可以优化容器内的数据存储,删除冗余的数据
  • 删除冗余数据:对于最小值的情况,j>i,aj<ai,那么之后窗口ai一定不会被选中,因此直接删除aj,由此我们得到一个单调递增的数组。此时,头部元素就是最小值
  • 最后使用队列方便我们的实现。
cpp 复制代码
int n,k,a[1000009];
deque<int>q;
void solve() {
	cin >> n >> k;
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	// min
	for (int i = 1; i <= n; i++) {
		if (i >= k + 1) {
			if (q.size()&&q.front() == a[i - k]) {
				q.pop_front();
			}
		}
		while (q.size() && q.back() > a[i]) {
			q.pop_back();
		}
		q.push_back(a[i]);
		if (i >= k) {
			cout << q.front() << " ";
		}
	}
	cout << endl;
	q.clear();
	// max
	for (int i = 1; i <= n; i++) {
		if (i >= k + 1) {
			if (q.size() && q.front() == a[i - k]) {
				q.pop_front();
			}
		}
		while (q.size() && q.back() < a[i]) {
			q.pop_back();
		}
		q.push_back(a[i]);
		if (i >= k) {
			cout << q.front() << " ";
		}
	}
	cout << endl;
}
相关推荐
Boom_Shu26 分钟前
长方形的关系
数据结构·c++·算法
Lsk_Smion2 小时前
力扣实训 _ [543].二叉树的直径 _ [23].合并K个升序列表
数据结构·算法·leetcode
ID_180079054734 小时前
淘宝商品详情数据接口深度解析:架构、鉴权、数据结构与实战
数据结构·架构
散峰而望5 小时前
【算法练习】算法练习精选:陶陶摘苹果(基础+升级)、Music Notes、字串变换,你能AC几道?
数据结构·c++·算法·leetcode·贪心算法·github·动态规划
凤凰院凶涛QAQ5 小时前
《Java版数据结构 & 集合类剖析》集合框架的封装设计与顺序表:“从 Iterable 到 ArrayList:集合框架的‘职业树“
java·开发语言·数据结构
8Qi85 小时前
LeetCode 148. 排序链表 —— 解法一:自顶向下递归(分治 + 归并)
数据结构·算法·leetcode·链表·递归·分治·归并
8Qi86 小时前
LeetCode 148. 排序链表 —— 解法二:自底向上归并(迭代,O(1) 空间)
数据结构·算法·leetcode·链表·归并·迭代
嘿黑嘿呦6 小时前
数据结构-图论-最小生成树
数据结构·算法·图论
欧米欧7 小时前
C++进阶数据结构之红黑树
数据结构
papership7 小时前
【入门级-数据结构-1、线性结构:链 表(单链表、双向链表、循环链表 )】
数据结构·算法·链表