首先,在这里给大家推荐一个网站,https://neooj.com:8082,这是我母校的网站

言归正传,题目描述

VIJOS-P1083 小白逛公园

Time Limit: 1 Sec  Memory Limit: 128 MB

Description

小新经常陪小白去公园玩,也就是所谓的遛狗啦…在小新家附近有一条“公园路”,路的一边从南到北依次排着n个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩了。一开始,小白就根据公园的风景给每个公园打了分。小新为了省事,每次遛狗的时候都会事先规定一个范围,小白只可以选择第a个和第b个公园之间(包括a、b两个公园)选择连续的一些公园玩。小白当然希望选出的公园的分数总和尽量高咯。同时,由于一些公园的景观会有所改变,所以,小白的打分也可能会有一些变化。 那么,就请你来帮小白选择公园吧。

Input

第一行,两个整数N和M,分别表示表示公园的数量和操作(遛狗或者改变打分)总数。接下来N行,每行一个整数,依次给出小白  开始时对公园的打分。接下来M行,每行三个整数。第一个整数K,1或2。K=1表示,小新要带小白出去玩,接下来的两个整数a和b给出了选择公园的范围(1≤a,b≤N);K=2表示,小白改变了对某个公园的打分,接下来的两个整数p和s,表示小白对第p个公园的打分变成了s(1≤p≤N)。其中,1≤N≤500  000,1≤M≤100  000,所有打分都是绝对值不超过1000的整数。

Output

小白每出去玩一次,都对应输出一行,只包含一个整数,表示小白可以选出的公园得分和的最大值。

Sample Input

5 3
1
2
-3
4
5
1 2 3
2 2 -1
1 2 3

Sample Output

2
-1

思路

  注:以下写的i代表的是一个左儿子的编号,i+1代表的是一个右儿子的编号,new代表的是他俩的父亲

  对于这道题目,是区间查询单点修改。又由于数据范围比较大,所以很容易分析出来用线段树进行维护。

  再分析应该怎么维护每一个区间,这道题应该维护四个状态,区间和sumi,从左开始最大连续和lefti,从右开始最大连续和righti,以及在这个区间中最大连续和maxi,具体如左图所示    

            

  对于建树,难的是怎么可以合并这些节点,如右图所示,每两个节点就可以这样合并,首先解释一下如何合并出来新的节点的左边连续最大和,可以看出,左面的和(sumi)加上右面的最大左边连续和(left(i+1)),也就是new_left1,但是这只是一种可能,还有一种可能是就是左边节点的左边最大连续和,这两个值取max,left(new)=max(new_left1,lefti);同样右边也是这样求,right(new)=max(new_right1,righti);求和好求,直接加就好啦,sum(new)=sumi+sum(i+1);求max就不太好求了,由于左儿子的右边最大和右儿子的左边最大可以合并,所以我又开了一个new_max1,max(new)=max(new_max1,maxi,max(i+1))。

  修改比较好修改,直接找到当前节点O(1)修改就好了,但要注意的是,修改过后我们需要重新维护一下当前点以及当前点以上与这个点有关的所有点,维护过程就是之前建树的过程,时间复杂度O(n*log(n));查询比较复杂,搜索区间和查询区间有三种情况(如下图)对于这三种情况进行以下讨论,会轻松的发现,当find_r>mid=(dfs_l+dfs_r)>>1时,应该便利右儿子,同理当find_l<=mid=(dfs_l+dfs_r)>>1时,应该便利左儿子,当查询区间完全包含搜索区间时,便没有必要查询,直接返回。当然本题是要查最大值,然而最大连续段有可能横跨多个区间,所以可以在找到的小节点上再建一棵线段树,最后直接输出新线段树的根节点信息就可以了,顺便说一下新的节点的建树和原本的建树一样,由于新的节点即答案的新线段树不好用数组写法,所以我用了动态开点线段树,而基础线段树用的是数组写法。

   

 #include<stdio.h>
int left[];
int right[];
int sum[];
int max[];
int cnt;
int ans_left[];
int ans_right[];
int ans_sum[];
int ans_max[];
int ans_lson[];
int ans_rson[];
void swap(int &a,int &b)
{
int tmp=a;
a=b,b=tmp;
}
int max1(int a,int b)
{
if(a>b)
return a;
return b;
}
void preserve(int p)
{
max[p]=max1(max[p<<],max[(p<<)+]);
max[p]=max1(max[p],right[p<<]+left[(p<<)+]);
left[p]=max1(sum[p<<]+left[(p<<)+],left[p<<]);
right[p]=max1(sum[(p<<)+]+right[p<<],right[(p<<)+]);
sum[p]=sum[p<<]+sum[(p<<)+];
}
void build(int l,int r,int p)
{
if(l==r)
{
int a;
scanf("%d",&a);
left[p]=sum[p]=right[p]=max[p]=a;
return;
}
build(l,(l+r)>>,p<<);
build(((l+r)>>)+,r,(p<<)+);
preserve(p);
}
void update(int p,int l,int r,int place,int delta)
{
if(l==r)
{
left[p]=sum[p]=right[p]=max[p]=delta;
return;
}
int mid=(l+r)>>;
if(place<=mid)
update(p<<,l,mid,place,delta);
else
update((p<<)+,mid+,r,place,delta);
preserve(p);
}
void find(int p,int l,int r,int x,int y)
{
int place=++cnt;
ans_lson[place]=ans_rson[place]=ans_sum[place]=;
if(l>=x&&r<=y)
{
ans_max[place]=max[p];
ans_sum[place]=sum[p];
ans_left[place]=left[p];
ans_right[place]=right[p];
return;
}
int mid=(l+r)>>,times=;
if(x<=mid)
{
ans_lson[place]=cnt+;
find(p<<,l,mid,x,y);
times++;
ans_max[place]=ans_max[ans_lson[place]];
ans_sum[place]=ans_sum[ans_lson[place]];
ans_left[place]=ans_left[ans_lson[place]];
ans_right[place]=ans_right[ans_lson[place]];
}
if(y>mid)
{
ans_rson[place]=cnt+;
find((p<<)+,mid+,r,x,y);
times++;
ans_max[place]=ans_max[ans_rson[place]];
ans_sum[place]=ans_sum[ans_rson[place]];
ans_left[place]=ans_left[ans_rson[place]];
ans_right[place]=ans_right[ans_rson[place]];
}
if(times==)
{
int son_l=ans_lson[place];
int son_r=ans_rson[place];
ans_max[place]=max1(ans_max[son_l],ans_max[son_r]);
ans_max[place]=max1(ans_max[place],ans_right[son_l]+ans_left[son_r]);
ans_left[place]=max1(ans_sum[son_l]+ans_left[son_r],ans_left[son_l]);
ans_right[place]=max1(ans_sum[son_r]+ans_right[son_l],ans_right[son_r]);
ans_sum[place]=ans_sum[son_l]+ans_sum[son_r];
}
}
int main()
{
int n,m;
scanf("%d%d",&n,&m);
build(,n,);
for(int i=;i<=m;i++)
{
int kind,a,b;
scanf("%d%d%d",&kind,&a,&b);
if(kind==)
{
if(a>b)
swap(a,b);
for(int i=;i<=cnt;i++)
ans_max[i]=ans_right[i]=ans_left[i]=ans_sum[i]=(int)-1e9;
cnt=;
find(,,n,a,b);
printf("%d\n",ans_max[]);
}
else
update(,,n,a,b);
}
}

JDOJ-P1260 VIJOS-P1083 小白逛公园的更多相关文章

  1. [vijos P1083] 小白逛公园

    不知怎地竟有种错觉此题最近做过= =目测是类似的?那道题貌似是纯动归? 本来今晚想做两道题的,一道是本题,一道是P1653疯狂的方格取数或NOI08 Employee,看看现在的时间目测这个目标又达不 ...

  2. Vijos 1083 小白逛公园(线段树)

    线段树,每个结点维护区间内的最大值M,和sum,最大前缀和lm,最大后缀和rm. 若要求区间为[a,b],则答案max(此区间M,左儿子M,右儿子M,左儿子rm+右儿子lm). ----------- ...

  3. [vijos]1083小白逛公园<线段树>

    描述 小新经常陪小白去公园玩,也就是所谓的遛狗啦…在小新家附近有一条“公园路”,路的一边从南到北依次排着n个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩了. 一开始,小白就根据公园的风景给每个公 ...

  4. 【vijos】P1083 小白逛公园

    [算法]线段树 [题解] 学自:https://vijos.org/p/1083/solution(wang_yanheng的回答) 回溯时维护一段区间的以下域: sumL:从左端点起连续区间的最大和 ...

  5. Bzoj 1756: Vijos1083 小白逛公园 线段树

    1756: Vijos1083 小白逛公园 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1021  Solved: 326[Submit][Statu ...

  6. BZOJ 1756: Vijos1083 小白逛公园

    题目 1756: Vijos1083 小白逛公园 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 856  Solved: 264[Submit][Sta ...

  7. 线段树 || BZOJ1756: Vijos1083 小白逛公园 || P4513 小白逛公园

    题面:小白逛公园 题解: 对于线段树的每个节点除了普通线段树该维护的东西以外,额外维护lsum(与左端点相连的最大连续区间和).rsum(同理)和sum……就行了 代码: #include<cs ...

  8. 洛谷 P4513 小白逛公园-区间最大子段和-分治+线段树区间合并(单点更新、区间查询)

    P4513 小白逛公园 题目背景 小新经常陪小白去公园玩,也就是所谓的遛狗啦… 题目描述 在小新家附近有一条“公园路”,路的一边从南到北依次排着nn个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩 ...

  9. vijos1083:小白逛公园

    小白逛公园 描述 小新经常陪小白去公园玩,也就是所谓的遛狗啦…在小新家附近有一条“公园路”,路的一边从南到北依次排着n个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩了. 一开始,小白就根据公园的 ...

随机推荐

  1. web服务器,应用程序服务器,http服务器的区别

    WEB服务器.应用程序服务器.HTTP服务器有何区别?IIS.Apache.Tomcat.Weblogic.WebSphere都各属于哪种服务器? 这个概念很重要. Web服务器的基本功能就是提供We ...

  2. VFS四大对象之一 struct super_block

    linux虚拟文件系统四大对象: 1)超级块(super block) 2)索引节点(inode) 3)目录项(dentry) 4)文件对象(file) 现在先介绍第一个 一.super_block的 ...

  3. 聊一聊Javasript继承

      前前后后已经快写了2年左右javaScript,刚开始只是简单用于一些表单验证和操作dom节点,并没有深入的使用,随着渐渐的深入,开始不想去写重复的代码(懒的开始),从而写简单的继承,封装,抽象等 ...

  4. 【Keras】基于SegNet和U-Net的遥感图像语义分割

    上两个月参加了个比赛,做的是对遥感高清图像做语义分割,美其名曰"天空之眼".这两周数据挖掘课期末project我们组选的课题也是遥感图像的语义分割,所以刚好又把前段时间做的成果重新 ...

  5. Linux6.X图形界面如何打开终端以及如何将终端加入右键

    今天刚安装了一个centos 6.9图形界面的系统,安装完成后,鼠标右击没有打开终端的按钮,在网上查了一些资料,搞明白了,分享给大家. 在左上角菜单[Applications]--->[Syst ...

  6. 【android studio】 gradle配置成本地离线zip包

    http://blog.csdn.net/jingleye/article/details/52689260       关键词:distributionUrl修改成本地路径,android stud ...

  7. TI-RTOS 定时器的使用

    定时器 在TI-RTOS中属于内核的一部分,因此想了解它的使用还是要阅读Bios_User_Guide.pdf. 主要用到这么几个API, 加粗字体 更多的定义可以在 ..\packages\ti\s ...

  8. PHPStudy+PHPStorm下配置隐藏项目入口文件

    img { max-width: 100% } 默认情况下项目入口文件是站点根目录下index.php文件,一般程序启动时通过这个文件,定义文件路径,配置重要节点(比如是否开启调试模式),注册路由等, ...

  9. .26-浅析webpack源码之事件流make(1)

    compilation事件流中,依然只是针对细节步骤做事件流注入,代码流程如图: // apply => this-compilation // apply => compilation ...

  10. Spark算子--distinct

    distinct--Transformation类算子 代码示例