题意:就是相当于动规里面的求最大连续子串,不同的是,这里需要读入一个区间x,y,输出的区间 a,b 且x<=a<=b<=y,使得a b的连续子串最长,而且询问次数达到了10的五次方。

我是在大白书上看到这个题,一开始想简单了,就照着刘汝佳的思路写,线段树里只存了前缀最大和,后缀最大和,总最大和,但题目要求输出区间,明显的需要添加记录维护区间位置的,而且不止要存储 a,b,还要存pre suf,前缀 后缀的边界位置。我第一次写的时候意识到了,但是那个时候已经有点混乱了,我以为在查询的时候,根据最大和就能确定区间。。。结果真是。。。智商被碾压。

还有个令人心酸的地方,直接导致我WA了整整一版,。。。尼玛,我记得校队选拔赛的时候也是同样的问题卡了我最后一个简单的题目,就是我用的中间变量暂存最大和,我知道最大和肯定会超过int,所以之前就设置了ll,但尼玛我定义中间变量的时候,居然下意识还是用的int。。。。。检查了我N多遍才查出来。。尼玛太心酸了。。一定要注意细节

#include <iostream>
#include <cstdio>
#include <cstring>
#define Lson x<<1,l,mid
#define Rson (x<<1|1),mid+1,r
#define N 500030
using namespace std; struct node//线段树节点只需要记录前缀最右边间位置pre,类似的后缀suf,以及真正的解区间,vx,vy
{
int vx,vy;
int suf,pre;
} tree[N*];
long long s[N];
void getup(int x,int l,int r)
{
if (s[tree[x<<].pre]-s [l-]>=s[tree[x<<|].pre]-s[l-]){
tree[x].pre=tree[x<<].pre;
}
else{
tree[x].pre=tree[x<<|].pre;
}
if (s[r]-s[tree[x<<].suf-]>=s[r]-s[tree[x<<|].suf-]){
tree[x].suf=tree[x<<].suf;
}
else{
tree[x].suf=tree[x<<|].suf;
}
long long v1=s[tree[x<<].vy]-s[tree[x<<].vx-];
long long v2=s[tree[x<<|].vy]-s[tree[x<<|].vx-];
long long v0=s[tree[x<<|].pre]-s[tree[x<<].suf-];
if (v1>=v2){
tree[x].vx=tree[x<<].vx;
tree[x].vy=tree[x<<].vy;
}
else{
tree[x].vx=tree[x<<|].vx;
tree[x].vy=tree[x<<|].vy;
}
if (v0==s[tree[x].vy]-s[tree[x].vx-]){
if (tree[x<<].suf<tree[x].vx){
tree[x].vx=tree[x<<].suf;
tree[x].vy=tree[x<<|].pre;
}
else if (tree[x<<].suf==tree[x].vx && tree[x<<|].pre<tree[x].vy){
tree[x].vy=tree[x<<|].pre;
}
}
if (v0>s[tree[x].vy]-s[tree[x].vx-]){
tree[x].vx=tree[x<<].suf;
tree[x].vy=tree[x<<|].pre;
} }
void build(int x,int l,int r)
{
if (l==r)
{
tree[x].pre=tree[x].suf=tree[x].vy=tree[x].vx=l;
return;
}
int mid=(l+r)/;
build(Lson);
build(Rson);
getup(x,l,r);
}
node query(int st,int e,int x,int l,int r)
{
if (st<=tree[x].vx && tree[x].vy<=e){
return tree[x];
}
int mid=(l+r)/;
if (st>mid){
return query(st,e,Rson);
}
if (e<=mid){
return query(st,e,Lson);
}
node a=query(st,e,Lson);
node b=query(st,e,Rson);
node c;
if (s[a.vy]-s[a.vx-]>=s[b.vy]-s[b.vx-]){
c.vx=a.vx;
c.vy=a.vy;
}
else {
c.vx=b.vx;
c.vy=b.vy;
}
if (s[a.pre]-s[st-]>=s[b.pre]-s[st-]){
c.pre=a.pre;
}
else{
c.pre=b.pre;
}
if (s[e]-s[a.suf-]>=s[e]-s[b.suf-]){
c.suf=a.suf;
}
else{
c.suf=b.suf;
}
if (s[b.pre]-s[a.suf-]==s[c.vy]-s[c.vx-]){
if (a.suf<c.vx){
c.vx=a.suf;
c.vy=b.pre;
}
if (a.suf==c.vx && b.pre<c.vy){
c.vy=b.pre;
}
}
else
if (s[b.pre]-s[a.suf-]>s[c.vy]-s[c.vx-]){
c.vx=a.suf;
c.vy=b.pre;
}
return c;
}
int main()
{
int n,m;
int counts=;
while (scanf("%d%d",&n,&m)!=EOF)
{
int i,j,k;
memset(s,,sizeof s);
long long tt;
for (i=;i<=n;i++){
scanf("%lld",&tt);
s[i]=s[i-]+tt;
}
build(,,n);
printf("Case %d:\n",++counts);
for (i=;i<=m;i++){
int ss,e;
scanf("%d%d",&ss,&e);
node ans=query(ss,e,,,n); printf("%d %d\n",ans.vx,ans.vy);
}
}
return ;
}

经典线段树 UVALive 3938/UVA 1400的更多相关文章

  1. ZOJ - 1610 经典线段树染色问题

    这个是一个经典线段树染色问题,不过题目给的是左右左右坐标,即[0,3]包含0-1这一段 1-2这一段 2-3这一段,和传统的染色不太一样,不过其实也不用太着急. 我们把左边的坐标+1,即可,那么[0, ...

  2. pku 2777(经典线段树染色问题)

    Count Color Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 41202   Accepted: 12458 Des ...

  3. poj-2828 Buy Tickets(经典线段树)

    /* Buy Tickets Time Limit: 4000MS Memory Limit: 65536K Total Submissions: 10207 Accepted: 4919 Descr ...

  4. poj 2528 poster经典线段树+lazy+离散化

    #include<cstdio> #include<cstring> #include<algorithm> using namespace std; ; #def ...

  5. ZOJ1610(经典线段树涂色问题)

    Description Painting some colored segments on a line, some previously painted segments may be covere ...

  6. poj Hotel 线段树

    经典线段树的题. 每个节点存储的信息:左端点连续空房间的长度,右端点连续空房间长度,连续空房间的最大长度. 由于要求每次必须从尽量靠左边的位置进行居住,那么搜索时应尽量让区间起始位置更小: 1.如果当 ...

  7. 『zkw线段树及其简单运用』

    阅读本文前,请确保已经阅读并理解了如下两篇文章: 『线段树 Segment Tree』 『线段树简单运用』 引入 这是一种由\(THU-zkw\)大佬发明的数据结构,本质上是经典的线段树区间划分思想, ...

  8. 线段树模板hdu 1166:敌兵布阵

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  9. 线段树模板hdu 1754:I Hate It

    I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

随机推荐

  1. Hive 中的 order by, sort by, distribute by 与 cluster by

    Order By order by 会对输入做全排序, 因此只有一个Reducer(多个Reducer无法保证全局有序), 然而只有一个Reducer, 会导致当输入规模较大时, 消耗较长的计算时间. ...

  2. 吴裕雄--天生自然java开发常用类库学习笔记:LinkedList类

    import java.util.LinkedList ; public class LinkedListDemo01{ public static void main(String args[]){ ...

  3. Redis 详解 (三) redis的五大数据类型详细用法

    目录 1.string 数据类型 2.hash 数据类型 3.list 数据类型 4.set 数据类型 5.zset 数据类型 6.系统相关命令 7.key 相关命令 我们说 Redis 相对于 Me ...

  4. bzoj 3522: [Poi2014]Hotel

    呵呵,一开始天真的我以为求个 西格玛 C(??,3)就好了.. (题解:比枚举2个数的再多一个,,一样搞) #include <bits/stdc++.h> #define LL long ...

  5. R 数据框的操作

    1.插入一列 根据自带数据集beaver 进行操作,比如插入一列id. > colnames(beaver1) [1] "day" "time" &quo ...

  6. Git详细命令

    Git Guidegit的三种方式只在本地使用:将本地仓库上传到Github:下载GitHub上的仓库:1.只在本地使用在Git Bush上输入命令 mkdir git-demo-1 ——创建一个目录 ...

  7. eclipse中svn重新设置账户

    查看svn版本:windows > preference > Team > SVN 1.如果svn插件是svnkit版 只需找到.keyring文件,一般目录是:eclipse安装目 ...

  8. ES6 之 对象属性的可枚举性和遍历

    1.Object.getOwnPropertyDescriptor() 解释:获取对对象属性的描述对象. let obj = { foo: 123 }; console.log(Object.getO ...

  9. 干货分享:常见的留学生Essay写作逻辑结构

    任何一种类型的Essay写作都应遵循一种逻辑结构,Long Essay可能在不同部分依据情况使用不同的逻辑结构.本文将为大家分享六种常见留学生Essay写作逻辑结构,为方便阅读本文采用中英文对照方式. ...

  10. 怎么在一个servlet中实现多个功能 ?如何使一个Servlet处理多个请求?

    自学javaweb一直不知道一个servelt可以有多个功能!看了别人代码才知道这个可以有! 平时你建立servelt时候你会吧doget和dopost这两个勾上,要想实现多个功能,你不必要勾选dog ...