[bzoj4127]Abs_树链剖分_线段树
Abs bzoj-4127
题目大意:给定一棵数,支持链加和链上权值的绝对值的和。
注释:$1\le n,m \le 10^5$,$\delta \ge 0$,$|a_i|\le 10^8$。
想法:看完题,以为又是什么数据结构裸题。然后发现绝对值... ...卧槽?啥jb玩意儿?绝对值?这怎么加?开始的想法是维护一个do标记,表示这个区间有没有负值,如果没有直接懒标记,如果有,往下走的时候负数取出来,然后二分治... ...不想写,上网查的题解。我们先树链剖分之后建线段树。紧接着我们对于一段区间维护一个小信息:maxdown。表示这个点所代表的区间中的所有负值中的最大值的绝对值。我们发现所有的增量都是正的,所以每一个数的正负号只能变动一次并且只能从负号变成正号。所以在区间修改的时候如果这个区间有负值我们就暴力修改即可。
最后,附上丑陋的代码... ...
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 100010
#define lson l,mid,x<<1
#define rson mid+1,r,x<<1|1
using namespace std;
typedef long long ll;
const int inf=1<<30;
int a[N],head[N],to[N<<1],next[N<<1],cnt,fa[N],deep[N],sv[N],bl[N],pos[N],tot,v[N];
int n,val[N<<2],si[N<<2],add[N<<2];
ll sum[N<<2];
void addedge(int x,int y)
{
to[++cnt]=y,next[cnt]=head[x],head[x]=cnt;
}
void dfs1(int x)
{
sv[x]=1;
for(int i=head[x];i;i=next[i])
if(to[i]!=fa[x])
fa[to[i]]=x,deep[to[i]]=deep[x]+1,dfs1(to[i]),sv[x]+=sv[to[i]];
}
void dfs2(int x,int c)
{
int k=0;
bl[x]=c,pos[x]=++tot,v[tot]=a[x];
for(int i=head[x];i;i=next[i])
if(to[i]!=fa[x]&&sv[to[i]]>sv[k])
k=to[i];
if(k)
{
dfs2(k,c);
for(int i=head[x];i;i=next[i])
if(to[i]!=fa[x]&&to[i]!=k)
dfs2(to[i],to[i]);
}
}
void pushup(int x)
{
int l=x<<1,r=x<<1|1;
sum[x]=sum[l]+sum[r];
if(val[l]>0&&val[r]>0)val[x]=min(val[l],val[r]);
else if(val[l]>0)val[x]=val[l];
else if(val[r]>0)val[x]=val[r];
else val[x]=0;
si[x]=si[l]+si[r];
}
void pushdown(int x)
{
if(add[x])
{
int l=x<<1,r=x<<1|1;
sum[l]+=(ll)si[l]*add[x],val[l]-=add[x],add[l]+=add[x];
sum[r]+=(ll)si[r]*add[x],val[r]-=add[x],add[r]+=add[x];
add[x]=0;
}
}
void build(int l,int r,int x)
{
if(l==r)
{
if(v[l]<0) sum[x]=val[x]=-v[l],si[x]=-1;
else sum[x]=v[l],val[x]=0,si[x]=1;
return;
}
int mid=(l+r)>>1;
build(lson),build(rson);
pushup(x);
}
void update(int b,int e,int a,int l,int r,int x)
{
if(b<=l&&r<=e&&(val[x]<=0||val[x]>a)) sum[x]+=(ll)si[x]*a,val[x]-=a,add[x]+=a;
else if(l==r) sum[x]=a-sum[x],val[x]=0,si[x]=1;
else
{
pushdown(x);
int mid=(l+r)>>1;
if(b<=mid) update(b,e,a,lson);
if(e>mid) update(b,e,a,rson);
pushup(x);
}
}
ll query(int b,int e,int l,int r,int x)
{
if(b<=l&&r<=e)return sum[x];
pushdown(x);
int mid=(l+r)>>1;
ll ans=0;
if(b<=mid)ans+=query(b,e,lson);
if(e>mid)ans+=query(b,e,rson);
return ans;
}
void modify(int x,int y,int z)
{
while(bl[x]!=bl[y])
{
if(deep[bl[x]]<deep[bl[y]])swap(x,y);
update(pos[bl[x]],pos[x],z,1,n,1),x=fa[bl[x]];
}
if(deep[x]>deep[y])swap(x,y);
update(pos[x],pos[y],z,1,n,1);
}
ll solve(int x,int y)
{
ll ans=0;
while(bl[x]!=bl[y])
{
if(deep[bl[x]]<deep[bl[y]])swap(x,y);
ans+=query(pos[bl[x]],pos[x],1,n,1),x=fa[bl[x]];
}
if(deep[x]>deep[y])swap(x,y);
ans+=query(pos[x],pos[y],1,n,1);
return ans;
}
int main()
{
int m,opt,x,y,z;
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
for(int i=1;i<n;i++)scanf("%d%d",&x,&y),addedge(x,y),addedge(y,x);
dfs1(1),dfs2(1,1),build(1,n,1);
while(m--)
{
scanf("%d%d%d",&opt,&x,&y);
if(opt==1)scanf("%d",&z),modify(x,y,z);
else printf("%lld\n",solve(x,y));
}
return 0;
}
小结:注意题目中的数据范围,对于一些比较特殊的性质要发掘并利用。
[bzoj4127]Abs_树链剖分_线段树的更多相关文章
- [bzoj4196][Noi2015]软件包管理器_树链剖分_线段树
软件包管理器 bzoj-4196 Noi-2015 题目大意:Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件 ...
- [bzoj3694]最短路_树链剖分_线段树
最短路 bzoj-3694 题目大意:给你一个n个点m条边的无向图,源点为1,并且以点1为根给出最短路树.求对于2到n的每个点i,求最短路,要求不经过给出的最短路树上的1到i的路径上的最后一条边. 注 ...
- bzoj 3730: 震波 动态点分治_树链剖分_线段树
##### 题目描述 : 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着 ...
- bzoj 4372: 烁烁的游戏 动态点分治_树链剖分_线段树
[Submit][Status][Discuss] Description 背景:烁烁很喜欢爬树,这吓坏了树上的皮皮鼠. 题意: 给定一颗n个节点的树,边权均为1,初始树上没有皮皮鼠. 烁烁他每次会跳 ...
- [NOI2015]软件包管理器 树链剖分_线段树
没有太大难度,刷水有益健康 Code: // luogu-judger-enable-o2 #include <bits/stdc++.h> #define setIO(s) freope ...
- 洛谷 P2542 [AHOI2005]航线规划 树链剖分_线段树_时光倒流_离线
Code: #include <map> #include <cstdio> #include <algorithm> #include <cstring&g ...
- poj 3237 Tree(树链剖分,线段树)
Tree Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 7268 Accepted: 1969 Description ...
- bzoj 4034 [HAOI2015] T2(树链剖分,线段树)
4034: [HAOI2015]T2 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1536 Solved: 508[Submit][Status] ...
- bzoj 1036 [ZJOI2008]树的统计Count(树链剖分,线段树)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 10677 Solved: 4313[Submit ...
随机推荐
- hdu 1242(BFS+优先队列)
Rescue Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Subm ...
- Tool:ProcessOn
ylbtech-Tool:ProcessOn ProcessOn是一个面向垂直专业领域的作图工具和社交网络,成立于2011年6月并于2012年启动.ProcessOn将全球的专家顾问.咨询机构.BPM ...
- AMQP server localhost:5672 closed the connection. Check login credentials: Socket closed
2016-04-13 09:23:38.755 18850 INFO oslo.messaging._drivers.impl_rabbit [req-fafc8542-9403-4b5a-89d2- ...
- Elasticsearch 7.1.1 集群 + 配置身份验证
一.安装Elasticsearch 1.1 环境说明 Centos7.6 Elasticsearch7.1.1 #挂载数据盘 fdisk /dev/vdb n,p,,回车,回车,wq fdisk -l ...
- thinkphp 上传多张图片
tp3.23 没有找到同时上传多张图片 手册有讲过:http://www.kancloud.cn/manual/thinkphp/1876 其实可以通过,多张图片多次上传来到达效果 hmlt: < ...
- Halcon学习笔记之支持向量机(一)
例程:class_overlap_svm.hdev 说明:这个例程展示了如何用一个支持向量机来给一幅二维的图像进行分类.使用二维数据的原因是因为它可以很容易地联想成为区域和图像.本例程中使用了三个互相 ...
- MYSQL日期时间字符串互转
--MYSQL date_format(date,'%Y-%m-%d') -------------->oracle中的to_char(); 日期时间转字符串 --MYSQL str_to_da ...
- Spring Boot (14) 数据源配置原理
数据源配置源码 这里截取org.springframework.boot.autoconfigure.jdbc.DataSourceConfiguration的部分源码,主要介绍Tomcat和Hika ...
- Android5.1关机充电界面尺寸修改
Android5.1关机充电界面尺寸修改 因为项目的屏幕尺寸和一般的手机屏幕不一样,因此关机充电界面在设备上运行后严重变形,就需要自己修改这个界面了,废话不多说了,开打开打! 首先要说明这里是以And ...
- 2B课程笔记分享_StudyJams_2017
课程2B-创建交互式应用(下) 概述 课程2B的内容主要包括:使用变量来更新欲显示在屏幕上的内容,为按钮添加事件响应(联系XML属性与Java方法)逻辑等. 后续的课程会逐步深入地讲解使用Java开发 ...