POJ 2796 Feel Good 【单调栈】
传送门:http://poj.org/problem?id=2796
题意:给你一串数字,需要你求出(某个子区间乘以这段区间中的最小值)所得到的最大值
例子:
6
3 1 6 4 5 2
当L=3,R=5时这段区间总和为6+4+5=15,最小值为4,所以最后的结果为60.
思路:单调栈的板子题了。没学过单调栈的我只能现学了。单调栈分为单调递增栈,单调递减栈。而这个题呢,要用到单调递增栈。怎么维护单调栈呢?那就是在遇到一个元素大于栈顶元素时,就将该元素加入栈中。(假定即为区间最小值,区间左端点L=i,区间右端点R=i)如果遇到一个小于栈顶的元素时,则把栈里面的元素弹出来,同时记录该元素向左延申的最大长度,即更新区间左端点。在弹出栈顶元素时把这个要弹出栈的元素(栈顶元素)的右端点赋值给这个它的下一个元素(当前栈中次大元素),即更新区间右端点,同时更新答案。
#include<iostream>
#include<cstdio>
using namespace std;
const int maxn = 1e6 + 10;
struct node
{
int l;
int r;
long long num;
} s[maxn];
long long a[maxn];
long long sum[maxn];
long long ans;
int ans_l;
int ans_r;
int top;
int n;
void solve(int top)
{
int l = s[top].l;
int r = s[top].r;
//更新答案
if((sum[r] - sum[l - 1])*s[top].num >= ans)
{
ans = (sum[r] - sum[l - 1]) * s[top].num;
ans_r = r;
ans_l = l;
}
//向右延申
if(top > 0)
{
s[top - 1].r = s[top].r;
}
}
int main()
{
scanf("%d", &n);
for(int i = 1; i <= n; i++)
{
scanf("%I64d", &a[i]);
//记录前缀和
sum[i] = a[i] + sum[i - 1];
}
top = -1;
for(int i = 1; i <= n; i++)
{
//初始假定a[i]为区间最小值,左区间,右区间都为i
node v;
v.l = i;
v.r = i;
v.num = a[i];
//加入的元素大于栈顶元素向左延申
while(top != -1 && s[top].num >= a[i])
{
solve(top);
//一直向左更新 ,直到某一个元素小于加入的元素
v.l = s[top].l;
top--;
}
s[++top].l = v.l;
s[top].r = v.r;
s[top].num = a[i];
}
//单调栈中元素可能不为空
while(top != -1)
{
solve(top);
top--;
}
cout << ans << endl;
cout << ans_l << " " << ans_r << endl;
}
POJ 2796 Feel Good 【单调栈】的更多相关文章
- poj 2796 Feel Good单调栈
Feel Good Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 20408 Accepted: 5632 Case T ...
- poj 2796 Feel Good 单调栈区间问题
Feel Good 题意:给你一个非负整数数组,定义某个区间的参考值为:区间所有元素的和*区间最小元素.求该数组中的最大参考值以及对应的区间. 比如说有6个数3 1 6 4 5 2 最大参考值为6,4 ...
- POJ 3658 Artificial Lake (单调栈)
题意: 析:利用单调栈,维护一个单调递增的栈,首先在最低的平台开始,每次向两边进行扩展,寻找两边最低的,然后不断更新宽度. 代码如下: #pragma comment(linker, "/S ...
- poj 2559 Largest Rectangle(单调栈)
Largest Rectangle in a Histogram Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 26549 ...
- POJ - 2796 Feel Good 单调递增栈+前缀和
Feel Good Bill is developing a new mathematical theory for human emotions. His recent investigations ...
- POJ 3415 后缀数组+单调栈
题目大意: 给定A,B两种字符串,问他们当中的长度大于k的公共子串的个数有多少个 这道题目本身理解不难,将两个字符串合并后求出它的后缀数组 然后利用后缀数组求解答案 这里一开始看题解说要用栈的思想,觉 ...
- poj 2796 Feel Good 单调队列
Feel Good Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 8753 Accepted: 2367 Case Ti ...
- [poj 2796]单调栈
题目链接:http://poj.org/problem?id=2796 单调栈可以O(n)得到以每个位置为最小值,向左右最多扩展到哪里. #include<cstdio> #include ...
- POJ 2796:Feel Good(单调栈)
http://poj.org/problem?id=2796 题意:给出n个数,问一个区间里面最小的元素*这个区间元素的和的最大值是多少. 思路:只想到了O(n^2)的做法. 参考了http://ww ...
随机推荐
- VS Code使用.vue的v-for方法提示错误的解决办法
1.在使用v-for的时候在后面跟上:key="key" <div v-for="(value, key) in [1,2,3]" :key=" ...
- jquery 获取同级元素
$(".userinfo-three:eq(0)").css({ "width": winWidth * 300 / 1080, ...
- 【LeetCode】分发糖果
[问题]老师想给孩子们分发糖果,有 N 个孩子站成了一条直线,老师会根据每个孩子的表现,预先给他们评分. 你需要按照以下要求,帮助老师给这些孩子分发糖果: 每个孩子至少分配到 1 个糖果.相邻的孩子中 ...
- java课程之团队开发冲刺阶段2.2
一.总结昨天进度 1.单独实现静音功能,还没有进行整体整合 二.遇到的问题 1.一开始设计静音的思路有问题,所以在实现上有些许麻烦,一开始的想法是将这些音量直接设置为0就可以实现静音,但是在恢复响铃模 ...
- SQL的查询结果复制到Excel 带标题Head 有换行符导致换行错乱 的解决方案
将SQL查询到的结果保存为excel有很多方法,其中最简单的就是直接复制粘贴了 1.带Head的复制粘贴 1)先左击红色区域实现选择所有数据 2)随后右击选择Copy with Headers 再粘 ...
- Android自定义View——贝塞尔曲线实现水波纹效果
我们使用到的是Path类的quadTo(x1, y1, x2, y2)方法,属于二阶贝塞尔曲线,使用一张图来展示二阶贝塞尔曲线,这里的(x1,y1)是控制点,(x2,y2)是终止点,起始点默认是Pat ...
- 阅读笔记--java内存模型原理
在阅读本篇文章之前,我所理解的和上网了解到的java内存模型原理如下: 不同架构的物理计算机可以有不一样的内存模型,Java 虚拟机也有自己的内存模型.Java 虚拟机规范中试图定义一种 Java 内 ...
- Servlet基础(java,idea)
IDEA的Tomcat配置Web的项目创建以及Servlet简单运行 登录测试 用到jdbc的知识 在Customer的DAO层实现类中实现检查用户登录方法 package impl; import ...
- Linux学习20200215
- IE8Get请求中文不兼容:encodeURI的使用
IE8Get请求中文不兼容:encodeURI的使用 在开发过程中遇到在IE8下,请求出错. 后发现Get请求中含有中文字符. 使用js自带的encodeURI函数对中文进行编码,问题解决. enco ...