这道题貌似可以用滑动窗口或者单调栈做, 但是我都没有用到。

这道题要求连续子序列中和乘上最小值最大, 那么我们就可以求出每一个元素,

以它为最小值的的最大区间的值, 然后取max就ok了。那么怎么求呢?

我可以初始化出一个第一个小于当前元素的的元素的位置, 也就是说初始化出以当前元素为最小值的
左右端点, 求出之后枚举一遍就ok了。

那么关键是怎么求呢? 这里有一点扫描法的味道。

假设当前序列为1243 以求右端点为例, 右端点的位置也就是在右侧第一个小于当前元素的元素的位置。
那么如果序列是递增的, 如开始的123, 那么123任何一个元素的右端点还没有找到, 所以继续往右。

但是如果遍历到了最右边的3, 显然左边的4 的右端点的值就是这个3, 而1 2则不是。

那么也就是说, 找到第一个不是大于前一个元素的元素之后(非递增), 那么就可以从右往左更新值, 只要左边的元素

大于这个元素, 那么就可以更新。而且只要一个更新不了, 那么左边的肯定也更新不了, 因为左边更小。

但是我们会发现, 下一次更新值的时候, 之前更新到的值还会被再扫一遍, 很浪费时间。

所以我这里用到了一个链表的结构, 更新完就在链表中去掉这个元素, 那么之后往左遍历的时候就会快很多了。

所以一个元素只会被更新到一次, 复杂度是O(n)的。

然后注意如果右边没有比自己小的元素的话, 那么就值就设置为最大右端点+1。

这样到后来算总和的时候,区间的右端点是这个数组的值-1(因为不包括右侧第一个小于当前元素的元素的位置)
那么这个时候就刚好是右半边整个区间了。
左端点类似

另外要到poj 2796去提交, UVa数据有问题。(我因此WA了好久)

#include<cstdio>
#include<cstring>
#include<queue>
#include<set>
#include<functional>
#include<algorithm>
#define REP(i, a, b) for(int i = (a); i < (b); i++)
using namespace std; const int MAXN = 112345;
typedef long long ll;
int r[MAXN], l[MAXN], a[MAXN], n;
int pre[MAXN], aft[MAXN], ans_l, ans_r;
ll sum[MAXN], ans; int main()
{
while(~scanf("%d", &n))
{
sum[0] = 0;
REP(i, 0, n)
{
scanf("%d", &a[i]);
sum[i] = (i == 0 ? 0 : sum[i-1]) + a[i];
pre[i] = i - 1; aft[i] = i + 1;
l[i] = r[i] = 0;
} REP(i, 1, n) //求右边第一个小于当前元素的元素的位置
if(a[i] < a[i-1])
{
int pos = i - 1;
while(pos >= 0 && a[pos] > a[i])
{
r[pos] = i;
pos = pre[pos];
pre[i] = pos;
}
}
REP(i, 0, n) if(r[i] == 0) r[i] = n; for(int i = n - 2; i >= 0; i--) //求左边第一个小于当前元素的元素的位置
if(a[i] < a[i+1])
{
int pos = i + 1;
while(pos < n && a[pos] > a[i])
{
l[pos] = i;
pos = aft[pos];
aft[i] = pos;
}
}
REP(i, 0, n) if(l[i] == 0) l[i] = -1; ans = -1;
REP(i, 0, n)
{
ll L = l[i] == -1 ? 0 : sum[l[i]];
ll R = sum[r[i]-1];
ll t = a[i] * (R - L);
if(ans < t)
{
ans = t;
ans_l = l[i]+2;
ans_r = r[i];
}
} printf("%lld\n%d %d\n", ans, ans_l, ans_r);
} return 0;
}

紫书 习题8-18 UVa 11536 (扫描法)的更多相关文章

  1. 紫书 习题 11-9 UVa 12549 (二分图最小点覆盖)

    用到了二分图的一些性质, 最大匹配数=最小点覆盖 貌似在白书上有讲 还不是很懂, 自己看着别人的博客用网络流写了一遍 反正以后学白书应该会系统学二分图的,紫书上没讲深. 目前就这样吧. #includ ...

  2. 紫书 习题 11-8 UVa 1663 (最大流求二分图最大基数匹配)

    很奇怪, 看到网上用的都是匈牙利算法求最大基数匹配 紫书上压根没讲这个算法, 而是用最大流求的. 难道是因为第一个人用匈牙利算法然后其他所有的博客都是看这个博客的吗? 很有可能-- 回归正题. 题目中 ...

  3. 紫书 习题8-12 UVa 1153(贪心)

    本来以为这道题是考不相交区间, 结果还专门复习了一遍前面写的, 然后发现这道题的区间是不是 固定的, 是在一个范围内"滑动的", 只要右端点不超过截止时间就ok. 然后我就先考虑有 ...

  4. 紫书 习题8-7 UVa 11925(构造法, 不需逆向)

    这道题的意思紫书上是错误的-- 难怪一开始我非常奇怪为什么第二个样例输出的是2, 按照紫书上的意思应该是22 然后就不管了,先写, 然后就WA了. 然后看了https://blog.csdn.net/ ...

  5. 紫书 习题 8-17 UVa 11536 (滑动窗口)

    这道题说连续子序列, 马上就想到滑动窗口. 注意窗口里面的元素中小于等于k的才是有效元素.记录窗口里面有效元素的个数, 满足了之后开始 缩短窗口, 如果左端点不是有效元素或者即使窗口中存在这个元素的个 ...

  6. 紫书 习题 11-10 UVa 12264 (二分答案+最大流)

    书上写的是UVa 12011, 实际上是 12264 参考了https://blog.csdn.net/xl2015190026/article/details/51902823 这道题就是求出一种最 ...

  7. 紫书 习题 11-17 UVa 1670 (图论构造)

    一开始要符合题目条件, 那么肯定没有任何一个点是孤立的, 也就是说没有点的度数是1 所以我就想让度数是1的叶子节点相互连起来.然后WA 然后看这哥们的博客 https://blog.csdn.net/ ...

  8. 紫书 习题 8-21 UVa 1621 (问题分析方法)

    知道是构造法但是想了挺久没有什么思路. 然后去找博客竟然只有一篇!!https://blog.csdn.net/no_name233/article/details/51909300 然后博客里面又说 ...

  9. 紫书 习题 10-13 UVa 11526(打表找规律+分步枚举)

    首先看这道题目,我预感商数肯定是有规律的排列的,于是我打表找一下规律 100 / 1 = 100 100 / 2 = 50  100 / 3 = 33  100 / 4 = 25  100 / 5 = ...

随机推荐

  1. IOS - JSON数据解析 小3种方法

    [manager GET:serverURL parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject ...

  2. vue2 在methods 中无法获取this对象

    在methods中使用箭头函数无法获取this ExamName:()=> { console.log(this);} 这样就行了: ExamName:function() { console. ...

  3. UNIX系统高级编程——第四章-文件和目录-总结

    文件系统: 以UNIX系统V文件系统为例: 磁盘分为区,每个分区都有自己的文件系统: ​ i节点是固定长度的记录项,包含了文件的相关信息.目录项包含文件名和i节点号.stat结构中除文件名和i节点编号 ...

  4. Android开发新手HelloWorld解析

    首先看这个 HelloWorld 类. Java代码public class HelloWorld extends Activity {       /** Called when the activ ...

  5. 新手学python-Day3-模块

    模块就是引入别人写的,官方写的工具库,就像扳手,钳子,电锯

  6. HDU3001 Traveling (状压dp+三进制+Tsp问题总结)

    (1)这道题最多可以走两次,所以有0, 1, 2三种状态,所以我们要用三进制 如果要用三进制,就要自己初始化两个数组, 一个是3的n次方,一个是三进制数的第几位的数字是什么 void init() { ...

  7. ucore_lab0

    一直想好好学习一下操作系统课程,去一个Mooc网站上找了一门操作系统的课程.这便是里面的配套实验. 实验指导:点这里 lab0主要是准备相关的操作环境.课程推荐使用qemu作为硬件模拟器,推荐运行环境 ...

  8. mybatis中根据日期模糊查询

    首先设置起始日期startDate和结束日期endDate,数据库中日期字段为achive_time,表名为dos_dossier<select id="getDossiers&quo ...

  9. 洛谷 P2071 座位安排 seat.cpp/c/pas

    P2071 座位安排 seat.cpp/c/pas 题目背景 公元二零一四年四月十七日,小明参加了省赛,在一路上,他遇到了许多问题,请你帮他解决. 题目描述 已知车上有N排座位,有N*2个人参加省赛, ...

  10. Tokyo Tyrant(TTServer)系列(二)-启动參数和配置

    启动參数介绍         ttserver命令能够启动一个数据库实例.由于数据库已经实现了Tokyo Cabinet的抽象API,所以能够在启动的时候指定数据库的配置类型. 支持的数据库类型有: ...