hdu 3966 Aragorn's Story(树链剖分+树状数组/线段树)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3966
题意:
给出一棵树,并给定各个点权的值,然后有3种操作:
I C1 C2 K: 把C1与C2的路径上的所有点权值加上K
D C1 C2 K:把C1与C2的路径上的所有点权值减去K
Q C:查询节点编号为C的权值
分析:
典型的树链剖分,对节点进行操作,可以用树状数组或者线段树。
树链剖分+树状数组:
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring> using namespace std;
const int maxn = ; struct Edge
{
int to,next;
}edge[maxn*];
int head[maxn],cnt,num;
int a[maxn],n,m,p,c[maxn];
int size[maxn],top[maxn],id[maxn],fa[maxn],son[maxn],dep[maxn]; void init()
{
memset(head,-,sizeof(head));
memset(son,-,sizeof(son));
memset(c,,sizeof(c));
cnt=;
num=;
}
void addedge(int u,int v)
{
edge[cnt].to = v;
edge[cnt].next = head[u];
head[u] = cnt++;
} void dfs_1(int u,int f,int d)
{
dep[u]=d;
size[u]=;
fa[u]=f;
for(int i=head[u];i!=-;i=edge[i].next)
{
int v=edge[i].to;
if(v==f)
continue;
dfs_1(v,u,d+);
size[u]+=size[v];
if(son[u]==-||size[son[u]]<size[v])
son[u]=v;
}
} void dfs_2(int u,int tp)
{
top[u] = tp;
id[u] = ++num;
if(son[u]!=-)
dfs_2(son[u],tp);
for(int i=head[u];i!=-;i=edge[i].next)
{
int v=edge[i].to;
if(v==fa[u]||v==son[u])
continue;
dfs_2(v,v);
}
} int lowbit(int x)
{
return x&-x;
} int sum(int x)
{
int res=;
while(x>)
{
res+=c[x];
x-=lowbit(x);
}
return res;
} void add(int x,int d)
{
while(x<=n)
{
c[x]+=d;
x+=lowbit(x);
}
} void change(int u,int v,int val)
{
int tp1=top[u],tp2=top[v];
while(tp1!=tp2)
{
if(dep[tp1]<dep[tp2])
{
swap(tp1,tp2);
swap(u,v);
}
add(id[tp1],val);
add(id[u]+,-val);
u=fa[tp1];
tp1=top[u];
}
if(dep[u]>dep[v])
swap(u,v);
add(id[u],val);
add(id[v]+,-val);
} int main()
{
while(~scanf("%d%d%d",&n,&m,&p))
{
init();
for(int i=;i<=n;i++)
scanf("%d",&a[i]);
for(int i=;i<=m;i++)
{
int u,v;
scanf("%d%d",&u,&v);
addedge(u,v);
addedge(v,u);
}
dfs_1(,,);
dfs_2(,);
for(int i=;i<=n;i++)
{
add(id[i],a[i]);
add(id[i]+,-a[i]);
}
char s[];
int c1,c2,k,c;
for(int i=;i<p;i++)
{
scanf("%s",s);
if(s[]=='I')
{
scanf("%d%d%d",&c1,&c2,&k);
change(c1,c2,k);
}
if(s[]=='D')
{
scanf("%d%d%d",&c1,&c2,&k);
change(c1,c2,-k);
}
if(s[]=='Q')
{
scanf("%d",&c);
cout<<sum(id[c])<<endl;
}
}
}
return ;
}
树链剖分+线段树:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int maxn = ;
int n,m,p;
int val[maxn],a[maxn];
struct Edge
{
int to,next;
}edge[maxn*];
int head[maxn],add[maxn*];
int dep[maxn],fa[maxn],size[maxn],son[maxn],top[maxn],id[maxn];
int cnt,num;
void init()
{
memset(head,-,sizeof(head));
memset(son,-,sizeof(son));
memset(add,,sizeof(add));
cnt=;
num=;
} void addedge(int u,int v)
{
edge[cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt++;
} void dfs_1(int u,int f,int d)
{
dep[u]=d;
size[u]=;
fa[u]=f;
for(int i=head[u];i!=-;i=edge[i].next)
{
int v=edge[i].to;
if(v==f)
continue;
dfs_1(v,u,d+);
size[u]+=size[v];
if(son[u]==-||size[son[u]]<size[v])
son[u]=v;
}
} void dfs_2(int u,int tp)
{
top[u] = tp;
id[u] = ++num;
if(son[u]!=-)
dfs_2(son[u],tp);
for(int i=head[u];i!=-;i=edge[i].next)
{
int v = edge[i].to;
if(v == fa[u] ||v == son[u])
continue;
dfs_2(v,v);
}
} struct Tree
{
int left,right;
int sum;
}tree[maxn*]; void pushup(int i)
{
tree[i].sum = tree[i*].sum + tree[i*+].sum;
} void build(int i,int begin,int end)
{
tree[i].left=begin;
tree[i].right=end;
if(begin==end)
{
tree[i].sum=val[begin];
return;
}
int mid=(begin+end)/;
build(i*,begin,mid);
build(i*+,mid+,end);
pushup(i);
} void pushdown(int i)
{
if(add[i])
{
add[i*] += add[i];
add[i*+] += add[i];
int mid=(tree[i].left+tree[i].right)/;
tree[i*].sum += add[i]*(mid-tree[i].left+);
tree[i*+].sum += add[i]*(tree[i].right-mid);
add[i]=;
}
} void update(int i,int begin,int end,int value)
{
if(tree[i].left>=begin&&tree[i].right<=end)
{
add[i]+=value;
tree[i].sum+=value*(tree[i].right-tree[i].left+);
return;
}
pushdown(i);
int mid=(tree[i].left+tree[i].right)/;
if(mid>=begin)
update(i*,begin,end,value);
if(mid<end)
update(i*+,begin,end,value);
pushup(i);
} void change(int u,int v,int value)
{
int tp1=top[u],tp2=top[v];
while(tp1!=tp2)
{
if(dep[tp1]<dep[tp2])
{
swap(tp1,tp2);
swap(u,v);
}
update(,id[tp1],id[u],value);
u = fa[tp1];
tp1 = top[u];
}
if(dep[u]>dep[v])
swap(u,v);
update(,id[u],id[v],value);
} long long query(int i,int begin,int end)
{
if(tree[i].left>=begin&&tree[i].right<=end)
return tree[i].sum;
pushdown(i);
int mid=(tree[i].left+tree[i].right)/;
long long ans=;
if(mid>=begin)
ans+=query(i*,begin,end);
if(mid<end)
ans+=query(i*+,begin,end);
return ans;
} int main()
{
while(~scanf("%d%d%d",&n,&m,&p))
{
init();
for(int i=;i<=n;i++)
scanf("%d",&a[i]);
for(int i=;i<=m;i++)
{
int u,v;
scanf("%d%d",&u,&v);
addedge(u,v);
addedge(v,u);
}
dfs_1(,,);
dfs_2(,); for(int i=;i<=n;i++)
val[id[i]]=a[i];
build(,,n);
char s[];
int c1,c2,k,c;
for(int i=;i<p;i++)
{
scanf("%s",s);
if(s[]=='I')
{
scanf("%d%d%d",&c1,&c2,&k);
change(c1,c2,k);
}
if(s[]=='D')
{
scanf("%d%d%d",&c1,&c2,&k);
change(c1,c2,-k);
}
if(s[]=='Q')
{
scanf("%d",&c);
cout<<query(,id[c],id[c])<<endl;
}
}
}
return ;
}
hdu 3966 Aragorn's Story(树链剖分+树状数组/线段树)的更多相关文章
- Qtree3题解(树链剖分(伪)+线段树+set)
外话:最近洛谷加了好多好题啊...原题入口 这题好像是SPOJ的题,挺不错的.看没有题解还是来一篇... 题意: 很明显吧.. 题解: 我的做法十分的暴力:树链剖分(伪)+线段树+\(set\)... ...
- Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树)
Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树) Description 一棵树上有n个节点,编号分别 ...
- 【bzoj4999】This Problem Is Too Simple! 树链剖分+动态开点线段树
题目描述 给您一颗树,每个节点有个初始值. 现在支持以下两种操作: 1. C i x(0<=x<2^31) 表示将i节点的值改为x. 2. Q i j x(0<=x<2^31) ...
- [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...
- 洛谷P3313 [SDOI2014]旅行(树链剖分 动态开节点线段树)
题意 题目链接 Sol 树链剖分板子 + 动态开节点线段树板子 #include<bits/stdc++.h> #define Pair pair<int, int> #def ...
- 刷题总结——骑士的旅行(bzoj4336 树链剖分套权值线段树)
题目: Description 在一片古老的土地上,有一个繁荣的文明. 这片大地几乎被森林覆盖,有N座城坐落其中.巧合的是,这N座城由恰好N-1条双 向道路连接起来,使得任意两座城都是连通的.也就是说 ...
- BZOJ 3531 [Sdoi2014]旅行 树链剖分+动态开点线段树
题意 S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰. 为了方便,我们用 ...
- hdu3966 树链剖分点权模板+线段树区间更新/树状数组区间更新单点查询
点权树的模板题,另外发现树状数组也是可以区间更新的.. 注意在对链进行操作时方向不要搞错 线段树版本 #include<bits/stdc++.h> using namespace std ...
- 【BZOJ3531】[Sdoi2014]旅行 树链剖分+动态开点线段树
[BZOJ3531][Sdoi2014]旅行 Description S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天 ...
- bzoj3531——树链剖分+动态开点线段树
3531: [Sdoi2014]旅行 Time Limit: 20 Sec Memory Limit: 512 MB Description S国有N个城市,编号从1到N.城市间用N-1条双向道路连 ...
随机推荐
- Python面向对象编程(三)
封装 1.为什么要封装? 封装就是要把数据属性和方法的具体实现细节隐藏起来,只提供一个接口.封装可以不用关心对象是如何构建的 2.封装包括数据的封装和函数的封装,数据的封装是为了保护隐私,函数的封装是 ...
- 移动端JS事件、移动端框架
一.移动端的操作方式和PC端是不同的,移动端主要是用手指操作,所以有特殊的touch事件,touch事件包括如下几个事件: 1.手指放到屏幕上时触发 touchstart 2.手指放在屏幕上滑动式 ...
- [补] winpcap编程——EAPSOCKET实现校园网锐捷登录(mentohust)
EAP SOCKET Implement Mentohust 时间20161115 对于EAP协议不了解,可参考上一篇随笔. 通过抓包分析校园网的锐捷登录流程,我在上一篇随笔中实现了EAPSOCKET ...
- 提交到APPStore出现ERROR ITMS-90474
解决的方案是:在工程的targets--->General----->Develoment Info ------->Status BarStyle
- C# 代码规范和质量检查工具 StyleCop.Analyzers
简介 原来一直用 ReSharper 来进行代码质量检查,不过毕竟是收费的,所以想找个免费的可以推广给公司的同事也一起用.搜索了一下,找到了StyleCop,但是我在 VS 2015里安装 Style ...
- Android与NativeC传递数据不正确问题
操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Android studio 2.3.3 这两天一直在调试一个BUG,具体为通过 NativeC 来处理上层Android ...
- 测试环境-memcached安装与说明
一,下载memcached和libevent安装包: 1,Memcache用到了libevent这个库用于Socket的处理,所以还需要安装libevent (这两个包百度上都能找到) 二,安装lib ...
- Java中实现十进制数转换为二进制的三种思路
Java中实现十进制数转换为二进制 第一种:除基倒取余法 这是最符合我们平时的数学逻辑思维的,即输入一个十进制数n,每次用n除以2,把余数记下来,再用商去除以2...依次循环,直到商为0结束,把余数倒 ...
- Presto向分区表快速插入数据时出现'target directory already exists'的原因
因为项目使用Presto作为ETL使用,需要将关系库中的数据导入到Hive中.目前关系库中的数据每天导入一次,在Hive中以天为间隔创建新的分区.思路是正确的,但是在使用的过程中,发现将少量关系库中的 ...
- React Native 系列(一) -- JS入门知识
前言 本系列是基于React Native版本号0.44.3写的,最初学习React Native的时候,完全没有接触过React和JS,本文的目的是为了给那些JS和React小白提供一个快速入门,让 ...