C++ 单调栈

C++单调栈模版代码:

cpp 复制代码
#include<iostream>
#include<vector>
#include<stack>
using namespace std;

/*单调栈模版*/
#define maxn 100001
#define inf 2000000000

template<typename T>
bool cmp(T a, T b) {
	return a > b;
}
// ans[i]代表从i往左走,找到的第一个满足条件的下标
template<typename T>
void findFirstMeetOnleft(int n, T h[], int ans[]) {
	stack<int> stk;
	h[0] = inf;
	stk.push(0);
	for (int i = 1; i <= n; ++i) {
		while(!cmp(h[stk.top()],h[i])) {
			stk.pop();
		}
		ans[i] = stk.top();
		stk.push(i);

	}
}

template<typename T>
void reverseArray(int n, T arr[]) {
	for (int i = 1; i <= n / 2; ++i) {
		T tmp = arr[i];
		arr[i] = arr[n + 1 - i];
		arr[n + 1 - i] = tmp;
	}
}

/*单调栈模版*/
int h[maxn], ans[maxn];

int main() {
	int n;
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		cin >> h[i];
	}
	findFirstMeetOnleft(n, h, ans);
	for (int i = 1; i <= n; ++i) {
		cout << ans[i] << " ";
	}
	cout << endl;
	return 0;
}

代码练习 1,伦太郎的等待,对应蓝桥云课 代码见下

cpp 复制代码
#include<iostream>
#include<vector>
#include<stack>
using namespace std;

/*单调栈模版*/
#define maxn 100001
#define inf 2000000000

template<typename T>
bool cmp(T a, T b) {
	return a >= b;
}
// ans[i]代表从i往左走,找到的第一个满足条件的下标
template<typename T>
void findFirstMeetOnleft(int n, T h[], int ans[]) {
	stack<int> stk;
	h[0] = inf;
	stk.push(0);
	for (int i = 1; i <= n; ++i) {
		while(!cmp(h[stk.top()],h[i])) {
			stk.pop();
		}
		ans[i] = stk.top();
		stk.push(i);

	}
}

template<typename T>
void reverseArray(int n, T arr[]) {
	for (int i = 1; i <= n / 2; ++i) {
		T tmp = arr[i];
		arr[i] = arr[n + 1 - i];
		arr[n + 1 - i] = tmp;
	}
}

/*单调栈模版*/
int h[maxn], ans[maxn];

int main() {
	int n;
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		cin >> h[i];
	}
	findFirstMeetOnleft(n, h, ans);
	int ret = 0;
	for (int i = 1; i <= n; ++i) {
		ret += i - ans[i] - 1;
	}
	cout << ret << endl;
	return 0;
}

代码练习2 对应蓝桥云课 百亿富翁 代码见下

cpp 复制代码
#include<iostream>
#include<vector>
#include<stack>
using namespace std;

/*单调栈模版*/
#define maxn 700001
#define inf 2000000000

template<typename T>
bool cmp(T a, T b) {
	return a > b;
}
// ans[i]代表从i往左走,找到的第一个满足条件的下标
template<typename T>
void findFirstMeetOnleft(int n, T h[], int ans[]) {
	stack<int> stk;
	h[0] = inf;
	stk.push(0);
	for (int i = 1; i <= n; ++i) {
		while(!cmp(h[stk.top()],h[i])) {
			stk.pop();
		}
		ans[i] = stk.top();
		stk.push(i);

	}
}

template<typename T>
void reverseArray(int n, T arr[]) {
	for (int i = 1; i <= n / 2; ++i) {
		T tmp = arr[i];
		arr[i] = arr[n + 1 - i];
		arr[n + 1 - i] = tmp;
	}
}

/*单调栈模版*/
int h[maxn], ans[maxn];

int main() {
	int n;
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		cin >> h[i];
	}
	findFirstMeetOnleft(n, h, ans);
	for (int i = 1; i <= n; ++i) {
		if (ans[i] == 0) ans[i] = -1;
		cout << ans[i] << ' ';
	}
	cout << endl;
	reverseArray(n, h);
	findFirstMeetOnleft(n, h, ans);
	reverseArray(n, ans);
	reverseArray(n, h);
	for (int i = 1; i <= n; ++i) {
		if (ans[i] == 0) ans[i] = -1;
		else ans[i] = (n + 1) - ans[i];
		cout << ans[i] << ' ';
	}
	cout << endl;
	return 0;
}

代码练习3 对应蓝桥云课 最大区间 代码见下

cpp 复制代码
#include<iostream>
#include<vector>
#include<stack>
using namespace std;

/*单调栈模版*/
#define maxn 300001
#define inf -1

template<typename T>
bool cmp(T a, T b) {
	return a < b;
}
// ans[i]代表从i往左走,找到的第一个满足条件的下标
template<typename T>
void findFirstMeetOnleft(int n, T h[], int ans[]) {
	stack<int> stk;
	h[0] = inf;
	stk.push(0);
	for (int i = 1; i <= n; ++i) {
		while(!cmp(h[stk.top()],h[i])) {
			stk.pop();
		}
		ans[i] = stk.top();
		stk.push(i);

	}
}

template<typename T>
void reverseArray(int n, T arr[]) {
	for (int i = 1; i <= n / 2; ++i) {
		T tmp = arr[i];
		arr[i] = arr[n + 1 - i];
		arr[n + 1 - i] = tmp;
	}
}

/*单调栈模版*/
int h[maxn], l[maxn], r[maxn];

int main() {
	int n;
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		cin >> h[i];
	}
	findFirstMeetOnleft(n, h, l);
	reverseArray(n, h);
	findFirstMeetOnleft(n, h, r);
	reverseArray(n, h);
	reverseArray(n, r);
	for (int i = 1; i <= n; ++i) {
		r[i] = (n + 1) - r[i];
	}
	long long max = 0;
	for (int i = 1; i <= n; ++i) {
		long long x = (r[i] - 1) - (l[i] + 1) + 1;
		x = x * h[i];
		if (x > max) {
			max = x;
		}
	}
	cout << max << endl;
	return 0;
}
相关推荐
Allen_LVyingbo3 分钟前
量子测量三部曲:投影测量、POVM 与坍缩之谜—从形式主义到物理图像
算法·性能优化·健康医疗·量子计算·空间计算
云栖梦泽6 分钟前
Linux内核与驱动:7.从应用层 lseek() 到驱动层 .llseek,Linux 字符设备偏移控制详解
linux·c++
qiqsevenqiqiqiqi8 分钟前
位运算 计算
算法
steins_甲乙14 分钟前
从0做一个小型内存泄露检测器(2): elf文件的动态链接
c++
甄心爱学习20 分钟前
【最优化】1-6章习题
人工智能·算法
charlie11451419120 分钟前
通用GUI编程技术——图形渲染实战(二十八)——图像格式与编解码:PNG/JPEG全掌握
开发语言·c++·windows·学习·图形渲染·win32
PD我是你的真爱粉20 分钟前
向量数据库原理与检索算法入门:ANN、HNSW、LSH、PQ 与相似度计算
数据库·人工智能·算法
汀、人工智能22 分钟前
[特殊字符] 第72课:杨辉三角
数据结构·算法·数据库架构·图论·bfs·杨辉三角
Ricky_Theseus32 分钟前
C++静态库
开发语言·c++
洛水水34 分钟前
【力扣100题】14.两数相加
c++·算法·leetcode