题目链接:https://vjudge.net/contest/146667#problem/C

题意:动态的求一个区间的最大连续和。

分析:

看上去可以RMQ去做,但是,当分成两个部分,原来的部分的解可能是跨越这两个区间的。原问题的解不能通过RMQ分成的两个部分的解而得到。

线段树:

线段树很早之前就有学习,那个时候只会套模板,而这个题目几乎就是线段树的一个理解应用。

就在刚刚之前提到的那个问题一样,可以利用线段树维护3个信息:

max_prefix(最大前缀和的标号)

max_suffix(最大后缀和的标号)

有了这两个信息,就可以跨区间找到丢失的信息了。

还有一个就是结果 max_sub(最大连续和的标号,其中是一个pair类型)

建树时:

建好左右子树后,递推max_prefix,max_suffix,这两个都不用跨区间,递推max_sub需要跨区间。

这里没有更新操作。

询问时:

同样分三种情况,左半边,右半边,跨区间。

跨区间这里,就用到了我们之前维护的最大前缀和标号,和最大后缀和标号。

那么如何得到最大前缀和标号,和最大后缀和标号呢?

同理:也是分区间查找,但是右半部分的左边的标号必须是L,左半部分简单一点,就是左孩子的最大前缀和的标号。

同理最大后缀和的标号。

 #include <bits/stdc++.h>

 using namespace std;

 const int maxn =  + ;
const int maxnode = + ;
typedef long long LL;
typedef pair<int,int> Interval; LL prefix_sum[maxn]; //求区间和
LL sum(int L,int R) {
return prefix_sum[R] - prefix_sum[L-];
} //求区间和
LL sum(Interval p) {
return sum(p.first,p.second);
} Interval better(Interval a,Interval b) {
if(sum(a)!=sum(b)) return sum(a) > sum(b) ? a : b;
return a<b? a:b;
} int qL,qR; struct IntervalTree
{
int max_prefix[maxnode]; //最大前缀和对应的标号
int max_suffix[maxnode]; //最大后缀和对应的标号
Interval max_sub[maxnode]; //最大连续和对应的起点和终点 void build(int o,int L,int R)
{
if(L==R)
{
max_prefix[o] = max_suffix[o] = L;
max_sub[o] = make_pair(L,L);
}
else
{
int M = L+(R-L)/;
int lc = o*,rc = o*+;
build(lc,L,M);
build(rc,M+,R); //递推最大前缀和
LL v1 = sum(L,max_prefix[lc]);
LL v2 = sum(L,max_prefix[rc]);
if(v1==v2) max_prefix[o] = min(max_prefix[lc],max_prefix[rc]);
else max_prefix[o] = v1 > v2 ? max_prefix[lc] : max_prefix[rc]; //递推最大后缀和
v1 = sum(max_suffix[lc],R);
v2 = sum(max_suffix[rc],R);
if(v1==v2) max_suffix[o] = min(max_suffix[lc],max_suffix[rc]);
else max_suffix[o] = v1 > v2 ? max_suffix[lc] : max_suffix[rc]; //递推最大连续和
max_sub[o] = better(max_sub[lc],max_sub[rc]);
max_sub[o] = better(max_sub[o],make_pair(max_suffix[lc],max_prefix[rc]));
}
} //求最大前缀和的那个区间
Interval query_prefix(int o,int L,int R)
{
if(max_prefix[o]<=qR) return make_pair(L,max_prefix[o]);
int M = L + (R-L)/;
int lc = o*,rc = o* + ;
if(qR<=M) return query_prefix(lc,L,M);
Interval i = query_prefix(rc,M+,R);
i.first = L;
return better(i,make_pair(L,max_prefix[lc]));
} Interval query_suffix(int o,int L,int R)
{
if(max_suffix[o]>=qL) return make_pair(max_suffix[o],R);
int M = L + (R-L) /;
int lc = o*,rc = o*+;
if(qL>M) return query_suffix(rc,M+,R);
Interval i = query_suffix(lc,L,M);
i.second = R;
return better(i,make_pair(max_suffix[rc],R)); } Interval query(int o,int L,int R)
{
if(qL<=L&&R<=qR) return max_sub[o];
int M = L + (R - L) /;
int lc = o*,rc = o*+;
if(qR<=M) return query(lc,L,M);
if(qL>M) return query(rc,M+,R);
Interval i1 = query_prefix(rc,M+,R);
Interval i2 = query_suffix(lc,L,M);
Interval i3 = better(query(lc,L,M),query(rc,M+,R)); //分开
return better(make_pair(i2.first,i1.second),i3); //跨区间
} }; IntervalTree tree; int main()
{
int n,q,a;
int cases = ;
while(scanf("%d%d",&n,&q)==)
{
prefix_sum[] = ;
for(int i=; i<n; i++)
{
scanf("%d",&a);
prefix_sum[i+] = prefix_sum[i] + a;
}
tree.build(,,n);
printf("Case %d:\n",cases++); while(q--) {
int L,R;
scanf("%d%d",&L,&R);
qL = L;
qR = R;
Interval ans = tree.query(,,n);
printf("%d %d\n",ans.first,ans.second);
} } return ;
}

LA 3938 动态最大连续和的更多相关文章

  1. LA 3938 动态最大连续和 线段树

    题目链接: https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show ...

  2. LA 3938 动态最大连续和(线段树)

    https://vjudge.net/problem/UVALive-3938 题意:给出一个长度为n的整数序列D,你的任务是对m个询问作出回答.对于询问(a,b),需要找到两个下标x和y,使得a≤x ...

  3. LA 3938 动态最大连续区间 线段树

    思路很清晰,实现很繁琐.分析过程可以参考LRJ,自己的总结晚些放. #include <cstdio> #include <cstring> #include <algo ...

  4. UVALive 3938 Ray, Pass me the dishes! (动态最大连续和)

    题意:求一个动态区间的最大连续和. 静态版本的O(n)算法显示不适用了,但是可以用线段树分治,因为一个连续和要在两边的区间,要么跨越两边,对于一个结点维护最大前缀和,后缀和,子区间连续和. 题目要求输 ...

  5. 2018年1月 mybatis+ 动态sql连续日期

    2018-01-111 .   如何在项目中快速查找对应的mapper.xml文件 从Controller入手,使用(Ctrl+鼠标左键)找到Service调用的方法,再找到底层方法,就能找到对应ma ...

  6. la 3938(未完成)

    题意:给出一个长度为n的整数序列D,你的任务是对m个询问作出回答.对于询问(a,b), 需要找到两个下标x和y,使得a≤x≤y≤b,并且Dx+Dx+1+...+Dy尽量大. 如果有多组满足条件的x和y ...

  7. 一个动态库连续注册的windows脚本regsvr32

    cmd ->for %1 in (%windir%\system32\*.dll) do regsvr32.exe /s %1

  8. LA 3938

    After doing Ray a great favor to collect sticks for Ray, Poor Neal becomes very hungry. In return fo ...

  9. 线段树(区间合并) LA 3989 "Ray, Pass me the dishes!"

    题目传送门 题意:动态最大连续子序列和,静态的题目 分析:nlogn的归并思想.线段树维护结点的三个信息,最大前缀和,最大后缀和,该区间的最大和的两个端点,然后答案是三个的better.书上用pair ...

随机推荐

  1. jQuery 全屏滚动插件 fullPage.js 参数说明

    fullPage.js 是一个基于 jQuery 的插件,它能够很方便.很轻松的制作出全屏网站,主要功能有: 支持鼠标滚动 支持前进后退和键盘控制 多个回调函数 支持手机.平板触摸事件 支持 CSS3 ...

  2. Launch iCar Scan Android Scanner Support Bluetooth X431 iDiag Update Version

    Autonumen.com release new Launch iCar Scan for Android,Launch iCarScan Bluetooth Scanner is update v ...

  3. Problem05 判断分数等级

    题目:利用条件运算符的嵌套来完成此题:学习成绩>=90分的同学用A表示,60-89分之间的用B表示,60分以下的用C表示. 程序分析:(a>b)?a:b这是条件运算符的基本例子. impo ...

  4. ubuntu 重新启动 导致 Namenode无法启动

    hadoop 单节点安装成功了之后,心情那是叫一个激动.不过后来不知道为什么系统相当的慢.于是就把系统重启了.但是不妙的是namenode各种无法启动. 原来的配置: core-site.xml &l ...

  5. CheckStyle unable to read from stream

    “我在对比了其他正确的ChechStyle文件之后,发现这个无法导入的文件的编码和正确文件的编码不一样,我的xml文档编码为ANSI,而导入正确的ChechStyle文件为UTF-8编码,在我将自己的 ...

  6. input类型为file改变默认按钮样式

    改变 input file 样式(input  文件域)是很多前端朋友经常遇到的头疼问题,今天推荐两种改变 input file 样式的两种常用方法: 方法一: <input type=&quo ...

  7. vue进阶 --- 实例演示

    这篇博客将通过一个实例来对vue构建项目的过程有一个了解. 主要用到的知识点如下所示: vue-router 2.0路由配置 router-view 和 router-link的使用 transiti ...

  8. static 和 final 和 static final

    众所周知,static 是静态修饰关键字:可以修饰变量,程序块,方法,类. 1.修饰变量. 得知:如果static修饰的是变量,则JVM会将将其分配在内存堆上,该变量就与对象无关,所有对该变量的引用都 ...

  9. 初识contiki(2.7版本)

    一个偶然的机会,我接触到了contiki这个家伙. Contiki 是一个开源的.高度可移植的.采用 C 语言开发的非常小型的嵌入式操作系统,针对小内存微控制器设计,适用于联网嵌入式系统和无线传感器网 ...

  10. 织梦DEDECMS {dede:arclist},{dede:list}获取附加表字段内容

    以前用织梦DEDECMS做二次开发时获取附加表字段内容都是通过runphp执行SQL查询获得,最近看了看手册,发现一个非常简便的方法. 用arclist调用于附加表字段的方法: 方法一: 要获取附加表 ...