3 5
题意:给n个数字,求出来他们区间的最大值,最大值计算方法:区间的所以数字中的最小数字*区间所有数字的和。
思路:单调栈处理的问题就是对每一个节点进行扩展的问题,这个题目要维护的是一个单调递减栈,即从栈顶元素到栈底元素,值是单调递减的,即栈顶元素的值始终是栈的最大值。然后每一个值有属于自己的区间,这个区间目的是为了记录之后的元素向前延伸的用处
向后延伸就靠从1到n扫描元素,(维护单调递减栈):
当扫描的元素大于栈顶元素时,直接入栈。
当扫描的元素等于栈顶元素时,不记录,只将区间延伸到后面。
当扫描的元素小于栈顶元素时,这时要计算栈内当前的值。因为扫描的元素时小于栈顶元素的,要求的是一个区间的最小值,所以栈内那些大于该元素的值你会发现没有用处了,只需要将它们的那些区间留下来就对了,这就是向前扩展。
拿题目的sample举例子:
3 1 6 4 5 2
一开始每一个数都有自己的区间:
3(1,1) 1(2,2) 6(3,3) 4(4,4) 5(5,5) 2(6,6) -1(7,7)后面加一个最小值,为了最后计算栈内元素使用。
先是3入栈。栈内元素 3(1,1)
1<3,首先计算一下栈内元素的值,记录下来。然后要把栈内大于1的全部弹出来,但是把它们的区间留下,栈内就变成了1(1,2)。实际上此时就会知道(1,2)这段区间之内的最小值是1。
6>1,直接入栈,栈内元素变为1(1,2),6(3,3)。
4<6,将6弹出,弹出之前计算值。然后栈内就变为1(1,2),4(3,4)。
5>4,直接入栈。栈内元素是1(1,2),4(3,4),5(5,5)。会发现因为5没有办法向前扩展了所以会知道5只能够在(5,5)的区间内最小,所以说站内元素是在自己区间的左端点与栈顶元素的右端点,这段区间之内满足着最小值的关系。1是在(1,5)这段区间内最小,4是在(3,5)这段区间内最小。这些值都会在碰到扫描的元素小于该元素时计算,记录下来,就是这样单调栈完成了对每一个元素进行左右扩展的目的。
2<5,2<4。要把5(5,5) 4(3,4)分别弹出,它们走之前要计算各自区间的值。
最后是-1,目的就是要将栈内所有元素弹出,计算每一个元素左右扩展的值。
#include
#include
#include
#include
#include
#include
#include
using namespace std;
typedef long long ll;
#define N 100005
int a[N], lef[N], stack[N], top;
long long sum[N];
int main()
{
ll ans = -1, tmp,n,r,l;
scanf("%lld", &n);
memset(sum, 0, sizeof(sum));
for (int i = 1; i <= n; i++)
{
scanf("%d", a + i);
sum[i] = sum[i - 1] + a[i];
}
a[++n] = -1;
top = 0;
for (int i = 1; i <= n; i++)
{
if (top == 0 || a[i] > a[stack[top - 1]])
{
stack[top++] = i;
lef[i] = i;
continue;
}
if (a[i] == a[stack[top - 1]])
continue;
while (top >= 1 && a[i] < a[stack[top - 1]])
{
--top;
tmp = 1LL*a[stack[top]] * (sum[i - 1] - sum[lef[stack[top]] - 1]);
if (tmp > ans)
{
l = lef[stack[top]];
r = i - 1;
ans = tmp;
}
}
lef[i] = lef[stack[top]];
stack[top++] = i;
}
printf("%lld\n%lld %lld\n", ans, l, r);
return 0;
}