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;
}
相关推荐
我在人间贩卖青春4 小时前
C++之数据类型的扩展
c++·字符串·数据类型
wangjialelele5 小时前
平衡二叉搜索树:AVL树和红黑树
java·c语言·开发语言·数据结构·c++·算法·深度优先
苏宸啊5 小时前
C++栈和队列
c++
xuxie995 小时前
day 21 双向链表以及循环链表
数据结构·链表
森G5 小时前
七、04ledc-sdk--------makefile有变化
linux·c语言·arm开发·c++·ubuntu
驱动探索者5 小时前
linux mailbox 学习
linux·学习·算法
ringking1235 小时前
autoware-1:安装环境cuda/cudnn/tensorRT库函数的判断
人工智能·算法·机器学习
大闲在人6 小时前
8. 供应链与制造过程术语:产能
算法·制造·供应链管理·智能制造·工业工程
橘颂TA6 小时前
【测试】高效浏览器操作:基础功能与优化设置大全
c++·功能测试·职场和发展·测试·web测试
一只小小的芙厨6 小时前
寒假集训笔记·以点为对象的树形DP
c++·算法