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_树链剖分_线段树的更多相关文章

  1. [bzoj4196][Noi2015]软件包管理器_树链剖分_线段树

    软件包管理器 bzoj-4196 Noi-2015 题目大意:Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件 ...

  2. [bzoj3694]最短路_树链剖分_线段树

    最短路 bzoj-3694 题目大意:给你一个n个点m条边的无向图,源点为1,并且以点1为根给出最短路树.求对于2到n的每个点i,求最短路,要求不经过给出的最短路树上的1到i的路径上的最后一条边. 注 ...

  3. bzoj 3730: 震波 动态点分治_树链剖分_线段树

    ##### 题目描述 : 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着 ...

  4. bzoj 4372: 烁烁的游戏 动态点分治_树链剖分_线段树

    [Submit][Status][Discuss] Description 背景:烁烁很喜欢爬树,这吓坏了树上的皮皮鼠. 题意: 给定一颗n个节点的树,边权均为1,初始树上没有皮皮鼠. 烁烁他每次会跳 ...

  5. [NOI2015]软件包管理器 树链剖分_线段树

    没有太大难度,刷水有益健康 Code: // luogu-judger-enable-o2 #include <bits/stdc++.h> #define setIO(s) freope ...

  6. 洛谷 P2542 [AHOI2005]航线规划 树链剖分_线段树_时光倒流_离线

    Code: #include <map> #include <cstdio> #include <algorithm> #include <cstring&g ...

  7. poj 3237 Tree(树链剖分,线段树)

    Tree Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 7268   Accepted: 1969 Description ...

  8. bzoj 4034 [HAOI2015] T2(树链剖分,线段树)

    4034: [HAOI2015]T2 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1536  Solved: 508[Submit][Status] ...

  9. bzoj 1036 [ZJOI2008]树的统计Count(树链剖分,线段树)

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 10677  Solved: 4313[Submit ...

随机推荐

  1. 【CTSC 2007】 数据备份

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1150 [算法] 首先,有一个很显然的结论 : 如果要使距离和最小,必须选择相邻的办公 ...

  2. bzoj 4198 [ Noi 2015 ] 荷马史诗 —— 哈夫曼编码(k叉哈夫曼树)

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4198 第一次写哈夫曼树!看了很多博客. 哈夫曼树 & 哈夫曼编码:https://w ...

  3. springMVC返回数据的四种方式

    转自:https://blog.csdn.net/itcats_cn/article/details/82119673 springMVC返回数据的四种方式:第一种,通过request.setAttr ...

  4. 第13课 SmartGit程序操作介绍

    http://www.syntevo.com/

  5. PCB MS SERVER 数据导出与导入操作步骤----使用第3方工具

    工作每天都与数据库打交道,经常会遇到一些需要将数据库中的数据导出来或将数据导入到数据库 而用微软数据库客户端自带的功能操作步骤好麻烦的,用过的大家都会有相同的感受吧. 微软客户端不好之处整理:这里吐槽 ...

  6. PCB MS SQL 小写转大写

    由于SQL Server允许为小写进入 ,导致数据库中存在小写,在数据集成到MES或ERP时报错,Oracle要求大写导致, 需转换为大写,可通过以下语句,查询所有小写数据,再更新.

  7. [SGU 199] Beautiful People

    [SGU 199] Beautiful People The most prestigious sports club in one city has exactly N members. Each ...

  8. 5.30dao-service-controller层,mybatis自动生成。(获取根据id主键获取指定详细数据)

    获取权限详细数据:(参考)                    1.controller:1.注入Servcie调用方法findConsumerById(参数是id);               ...

  9. var _this = this 是干什么的

    因为JS可以多层嵌套代码可能下面还可以再嵌一个方法引用this就会变成子方法控制的对象如果需要上级的对象在没有参数的情况下前面前提做了一个临时变量_this可以保存上级对象子方法中就可以用_this来 ...

  10. [2月1号] 努比亚全机型ROM贴 最全最新NubiaUI5.0 ROOT 极速体验

    前言 感谢在开发过程中mandfx和dgtl198312予以的帮助!本帖将整理所有Nubia手机的最新刷机包,还有少数机型未制作刷机包,需要的机油可以联系我制作recovery以及刷机包加群23722 ...