JDOJ-P1260 VIJOS-P1083 小白逛公园
首先,在这里给大家推荐一个网站,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
Sample Output
思路
注:以下写的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 小白逛公园的更多相关文章
- [vijos P1083] 小白逛公园
不知怎地竟有种错觉此题最近做过= =目测是类似的?那道题貌似是纯动归? 本来今晚想做两道题的,一道是本题,一道是P1653疯狂的方格取数或NOI08 Employee,看看现在的时间目测这个目标又达不 ...
- Vijos 1083 小白逛公园(线段树)
线段树,每个结点维护区间内的最大值M,和sum,最大前缀和lm,最大后缀和rm. 若要求区间为[a,b],则答案max(此区间M,左儿子M,右儿子M,左儿子rm+右儿子lm). ----------- ...
- [vijos]1083小白逛公园<线段树>
描述 小新经常陪小白去公园玩,也就是所谓的遛狗啦…在小新家附近有一条“公园路”,路的一边从南到北依次排着n个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩了. 一开始,小白就根据公园的风景给每个公 ...
- 【vijos】P1083 小白逛公园
[算法]线段树 [题解] 学自:https://vijos.org/p/1083/solution(wang_yanheng的回答) 回溯时维护一段区间的以下域: sumL:从左端点起连续区间的最大和 ...
- Bzoj 1756: Vijos1083 小白逛公园 线段树
1756: Vijos1083 小白逛公园 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1021 Solved: 326[Submit][Statu ...
- BZOJ 1756: Vijos1083 小白逛公园
题目 1756: Vijos1083 小白逛公园 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 856 Solved: 264[Submit][Sta ...
- 线段树 || BZOJ1756: Vijos1083 小白逛公园 || P4513 小白逛公园
题面:小白逛公园 题解: 对于线段树的每个节点除了普通线段树该维护的东西以外,额外维护lsum(与左端点相连的最大连续区间和).rsum(同理)和sum……就行了 代码: #include<cs ...
- 洛谷 P4513 小白逛公园-区间最大子段和-分治+线段树区间合并(单点更新、区间查询)
P4513 小白逛公园 题目背景 小新经常陪小白去公园玩,也就是所谓的遛狗啦… 题目描述 在小新家附近有一条“公园路”,路的一边从南到北依次排着nn个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩 ...
- vijos1083:小白逛公园
小白逛公园 描述 小新经常陪小白去公园玩,也就是所谓的遛狗啦…在小新家附近有一条“公园路”,路的一边从南到北依次排着n个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩了. 一开始,小白就根据公园的 ...
随机推荐
- .NET Core单文件发布静态编译AOT CoreRT
.NET Core单文件发布静态编译AOT CoreRT,将.NET Core应用打包成一个可执行文件并包含运行时. 支持Windows, MacOS and Linux x64 w/ RyuJIT ...
- .NET(c#) 移动开发平台 - Smobiler(1)
如果说基于.net的移动开发平台,目前比较流行的可能是xamarin了,不过除了这个,还有一个比xamarin更好用的国内的.net移动开发平台,smobiler,不用学习另外一套开发模式或者搭建复杂 ...
- 第四章 Windows图形界面-上
学习<Windows程序设计>记录 概念贴士: 1. 每个GUI应用程序至少应该创建一个窗口,称为主窗口,它作为用户与应用程序间的主界面来提供服务.大多数应用程序也直接或间接地创建其他窗口 ...
- 前端学习:html基础学习五
9.HTML表单设计(主要内容<form><input><select>标记) 表单标记 <form>...</form> <form ...
- sql万能密码
输入1'or'2这样就会引起sql注入,因为username=password admin adn admin,所以我们能够进去 必须要做好过滤措施
- [bzoj1706] [usaco2007 Nov]relays 奶牛接力跑
大概是叫倍增Floyd? 显然最多200个点...f[i][j][k]表示从j到k,走2^i步的最小路程.就随便转移了.. 查询的话就是把n二进制位上是1的那些都并起来. #include<cs ...
- STOI补番队互测#2
Round2轮到我出了>_<(目测总共10人参加 实际共七人) 具体情况: #1: KPM,360 #2:ccz181078,160 #3:child,150 可惜KPM没看到第一题样例里 ...
- [国嵌攻略][100][嵌入式Linux内核制作]
Linux内核制作步骤 1.清除原有配置 make distclean 2.配置内核 选择一个已有的配置文件简化配置 make menuconfig ARCH=arm 3.编译内核 ARCH指明处理器 ...
- putty 与winscp 区别
https://zhidao.baidu.com/question/377968180.html putty 与winscp 有什么区别, 装了 winscp 可以由 putty 替换么 ? 具体用法 ...
- dede后台出现 保存目录数据时失败,请检查你的输入资料是否存在问题
dede 5.7无法增加顶级/二级栏目,保存目录数据时失败,请检查你的输入资料是否存在问题!执行了SQL还是不行 解决档案:用正常可以添加栏目的,将E:\wamp\www\dededln\back(d ...