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;
}
相关推荐
ULTRA??2 小时前
插入排序算法实现(二分查找搜索版本)
c++·算法
Elias不吃糖2 小时前
LeetCode 71:简化 Unix 路径(Simplify Path)——栈 / vector
算法·leetcode·
sheeta19982 小时前
LeetCode 每日一题笔记 日期:2025.12.15 题目:2110.股票平滑下跌阶段的数目
笔记·算法·leetcode
郭涤生8 小时前
布隆过滤器
c++
智者知已应修善业9 小时前
【求中位数】2024-1-23
c语言·c++·经验分享·笔记·算法
9ilk9 小时前
【C++】--- 特殊类设计
开发语言·c++·后端
地平线开发者9 小时前
PTQ 量化数值范围与优化
算法·自动驾驶
sali-tec9 小时前
C# 基于halcon的视觉工作流-章68 深度学习-对象检测
开发语言·算法·计算机视觉·重构·c#
测试人社区-小明9 小时前
智能弹性伸缩算法在测试环境中的实践与验证
人工智能·测试工具·算法·机器学习·金融·机器人·量子计算
罗西的思考10 小时前
【Agent】MemOS 源码笔记---(5)---记忆分类
人工智能·深度学习·算法