模板题,无背景。
2019.12.12 更新数据,放宽时限,现在不再卡常了。
给出项数为 n n n 的整数数列 a 1 … n a_{1 \dots n} a1…n。
定义函数 f ( i ) f(i) f(i) 代表数列中第 i i i 个元素之后第一个大于 a i a_i ai 的元素的下标,即 f ( i ) = min i < j ≤ n , a j > a i { j } f(i)=\min_{i
试求出 f ( 1 … n ) f(1\dots n) f(1…n)。
第一行一个正整数 n n n。
第二行 n n n 个正整数 a 1 … n a_{1\dots n} a1…n。
一行 n n n 个整数 f ( 1 … n ) f(1\dots n) f(1…n) 的值。
5
1 4 2 3 5
2 5 4 5 0
【数据规模与约定】
对于 30 % 30\% 30% 的数据, n ≤ 100 n\leq 100 n≤100;
对于 60 % 60\% 60% 的数据, n ≤ 5 × 1 0 3 n\leq 5 \times 10^3 n≤5×103 ;
对于 100 % 100\% 100% 的数据, 1 ≤ n ≤ 3 × 1 0 6 1 \le n\leq 3\times 10^6 1≤n≤3×106, 1 ≤ a i ≤ 1 0 9 1\leq a_i\leq 10^9 1≤ai≤109。
考察单调栈,建议自己手推一波。
单调栈的关键在于找单调性和考虑什么时候栈中的元素没有贡献了。
单调栈的2个关键时间点:
假设现在栈里面有数字 1 1 1 ,现在考虑数字 4 4 4
先考虑入栈前: 1 < 4 1 < 4 1<4,所以 4 4 4 入栈后, 1 1 1 就没有贡献了,且 1 1 1 的 f ( i ) f(i) f(i) 就是 4 4 4 的下标,可以将 1 1 1 出栈;
再考虑入栈后: 4 > 2 4 > 2 4>2,所以 2 2 2 入栈后, 4 4 4 仍然没有找到它的 f ( i ) f(i) f(i) , 4 4 4 不能出栈,但是 2 2 2 也没有找到它的 f ( i ) f(i) f(i) ,所以 2 2 2 也不能出栈,必须入栈;
再考虑 3 3 3 入栈,因为 2 < 3 2 < 3 2<3,所以 3 3 3 是 2 2 2 的 f ( i ) f(i) f(i) ,且 3 3 3 入栈后, 2 2 2 就没有贡献了, 2 2 2 出栈, 3 3 3 入栈
综上,维护的栈是一个单调递减栈,维护方法是:当前准备入栈元素x
如果大于等于栈顶元素,则栈中小于x
的元素都出栈,然后x
入栈;如果小于栈顶元素,则x
直接入栈
#include
using namespace std;
const int N = 3e6 + 10;
int n;
int stk[N], a[N], f[N];
int top = -1;
int main() {
scanf("%d", &n);
for (int i = 1; i <= n; i++)
scanf("%d", &a[i]);
for (int i = 1; i <= n; i++) {
while (top != -1 && a[stk[top]] <= a[i]) {
f[stk[top]] = i;
top--;
}
if (top == -1)
f[i] = 0;
stk[++top] = i;
}
for (int i = 1; i <= n; i++)
printf("%d ", f[i]);
return 0;
}