shoi 魔法树
Harry Potter新学了一种魔法:可以改变树上的果子个数。满心欢喜的他找到了一个巨大的果树,来试验他的新法
术。这棵果树共有N个节点,其中节点0是根节点,每个节点u的父亲记为fa[u],保证有fa[u]<u。初始时,这棵果
树上的果子都被Dumbledore用魔法清除掉了,所以这个果树的每个节点都没有果子(即0个果子)不幸的是,Harry
的法术学得不到位,只能对树上一段路径的节点上的果子个数统一增加一定的数量。也就是说,Harry的魔法可以
这样描述:Add u v d,表示将节点u和v之间的路径上的所有节点的果子个数都加上d。接下来,为了方便检验Harr
y的魔法是否成功,你需要告诉他在释放魔法的过程中的一些有关果树的信息:Query u,表示当前果树中,以点u
为根的子树中,总共有多少个果子?
Input
第一行一个整数N(1≤N≤100000),表示果树的节点总数,节点以0,1,...,N-1标号,0一定代表根节点。
接下来N-1行,每行两个整数a,b(0≤a接下来是一个整数Q(1≤Q≤100000),表示共有Q次操作。
后面跟着Q行,每行是以下两种中的一种:
1. A u v d,表示将u到v的路径上的所有节点的果子数加上d;(0≤u,v≤N-1,0<d<100000)
2. Q u,表示询问以u为根的子树中的总果子数,注意包括u本身的。(0≤u≤N-1)
1≤N≤100000,1≤Q≤100000
Output
输出询问的答案。
Sample Input
4
0 1
1 2
2 3
4
A 1 3 1
Q 0
Q 1
Q 2
Sample Output
3
3
2
//记下dfs序,就知道子树的范围了
//对一段范围进行加值,lazy操作
#include <stdio.h>
#include <iostream>
#include <string.h>
#include <algorithm>
#include <stdlib.h>
#include <math.h>
#include <vector>
using namespace std; typedef long long ll;
const int N=100001; int e[N*2],nxt[N*2],head[N];
int dep[N],fa[N],siz[N],w[N],top[N],son[N],End[N]; struct A{
ll sum,tag;
}t[N*4]; int n,C=0,edge=0;
void add(int x,int y)
{
e[++edge]=y;
nxt[edge]=head[x];
head[x]=edge;
} void dfs1(int x,int f,int d)
{//求size,dep,son,fa
dep[x]=d;fa[x]=f;siz[x]=1;
int i,tp=0;
for(i=head[x];i;i=nxt[i])
if(e[i]!=fa[x])
{
dfs1(e[i],x,d+1);
siz[x]+=siz[e[i]];
if(siz[e[i]]>tp){tp=siz[e[i]];son[x]=e[i];}
}
} void dfs2(int x,int tp)
{//求dfs序,链的顶端节点
w[x]=++C;
top[x]=tp;
if(son[x])
dfs2(son[x],top[x]);
for(int i=head[x];i;i=nxt[i])
if(e[i]!=son[x]&&e[i]!=fa[x])
dfs2(e[i],e[i]);
End[x]=C;//x所在子树的最大dfs序
} void build(int x,int l,int r)
{//建立线段树
t[x].tag=t[x].sum=0;
if(l==r) return;
int M=l+r>>1;
build(x<<1,l,M);
build(x<<1|1,M+1,r);
} void pushdown(int x,int l,int r)
{//在线段树中下传标记
if(t[x].tag)
{
t[x<<1].tag+=t[x].tag;
t[x<<1|1].tag+=t[x].tag;
int M=l+r>>1;
t[x<<1].sum+=(M-l+1)*t[x].tag;
t[x<<1|1].sum+=(r-M)*t[x].tag;
t[x].tag=0;
}
} void update(int x,int l,int r,int ql,int qr,int v)
{//在线段树中更新[ql,qr]节点权值加v
if(ql<=l&&qr>=r)
{
t[x].sum+=(r-l+1)*v;
t[x].tag+=v;
return;
}
pushdown(x,l,r);
int M=l+r>>1;
if(ql<=M)
update(x<<1,l,M,ql,qr,v);
if(qr>M)
update(x<<1|1,M+1,r,ql,qr,v);
t[x].sum=t[x<<1].sum+t[x<<1|1].sum;
} ll query(int x,int l,int r,int ql,int qr)
{//在线段树中查询[ql,qr]的权值和
if(ql<=l&&qr>=r) return t[x].sum;
pushdown(x,l,r);
int M=l+r>>1;ll sum=0;
if(ql<=M) sum+=query(x<<1,l,M,ql,qr);
if(qr>M) sum+=query(x<<1|1,M+1,r,ql,qr);
return sum;
} void upd(int x,int y,int z)
{//修改操作,见课件
int f1=top[x],f2=top[y],tmp;
while(f1!=f2)
{
if(dep[f1]<dep[f2])
{
tmp=f1;f1=f2;f2=tmp;
tmp=x;x=y;y=tmp;
}
update(1,1,n,w[f1],w[x],z);
x=fa[f1];
f1=top[x];
}
if(dep[x]<dep[y]){tmp=x;x=y;y=tmp;}
update(1,1,n,w[y],w[x],z); //注意这一句,本题是对点权进行操作
} int main()
{
int i,x,y,m,z;char c;
scanf("%d",&n);
for(i=1;i<n;i++)
{
scanf("%d %d",&x,&y);x++;y++;
add(x,y);
add(y,x);
}
dfs1(1,-1,1);
dfs2(1,1);
build(1,1,n);
scanf("%d",&m);
for(i=1;i<=m;i++)
{
scanf("\n%c",&c);
if(c=='A')
{
scanf("%d %d %d",&x,&y,&z);x++;y++;
upd(x,y,z);
}
else
{
scanf("%d",&x);//统计以x为根的子树的权值
x++;
printf("%lld\n",query(1,1,n,w[x],End[x]));
}
}
return 0;
}
shoi 魔法树的更多相关文章
- noip模拟赛(一)魔法树
魔法树 (mahou.pas/c/cpp) [问题描述] 魔法使moreD在研究一棵魔法树. 魔法树顾名思义,这货是一棵树,奇葩的是魔法树上的每一条边都拥有一个魔法属性,如果不那么奇葩就不是moreD ...
- [SHOI 2012] 魔法树
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=2836 [算法] 树链剖分 时间复杂度 : O(NlogN ^ 2) [代码] #in ...
- 洛谷 3833 SHOI 2012 魔法树
[题解] 树链剖分模板题.. #include<cstdio> #include<algorithm> #include<queue> #define N 5000 ...
- [bzoj2836] 魔法树
俩操作:增加路径上的点的权值.查询子树的权值和. 想了想似乎只能树链剖分了..好久没写链剖+数据结构了TAT 一开始没开LL炸了一发(明明有想到的..我果然是傻逼= = #include<cst ...
- P3833 [SHOI2012]魔法树
思路 树剖板子 注意给出点的编号是从零开始的 代码 #include <cstdio> #include <algorithm> #include <cstring> ...
- [洛谷P3833][SHOI2012]魔法树
题目大意:给一棵树,路径加,子树求和 题解:树剖 卡点:无 C++ Code: #include <cstdio> #include <iostream> #define ma ...
- 【树链剖分】【dfs序】【线段树】bzoj2836 魔法树
这道题告诉我们:树链剖分的重标号就是dfs序. #include<cstdio> #include<algorithm> using namespace std; #defin ...
- 树链剖分【洛谷P3833】 [SHOI2012]魔法树
P3833 [SHOI2012]魔法树 题目描述 Harry Potter 新学了一种魔法:可以让改变树上的果子个数.满心欢喜的他找到了一个巨大的果树,来试验他的新法术. 这棵果树共有N个节点,其中节 ...
- 树链剖分【P3833】 [SHOI2012]魔法树
Description Harry Potter 新学了一种魔法:可以让改变树上的果子个数.满心欢喜的他找到了一个巨大的果树,来试验他的新法术. 这棵果树共有N个节点,其中节点0是根节点,每个节点u的 ...
随机推荐
- oracle高水位降低法
1.什么是高水位?(high water mark 简称:HWM) 所有的oracle段(segments,在此,为了理解方便,建议把segment作为表的一个同义词)都有一个在段内存放数据的 ...
- 火车采集用到的access查询命令小结
#For zencart #图片网址路径替换 UPDATE Content SET v_products_image=replace(v_products_image, '<img src=&q ...
- 探索super()的执行顺序和__mro__方法
class Base(object): def func(self): print('Base.func') class Foo(Base): def func(self): # 方式一:根据mro的 ...
- JVM metaspace元空间
元空间的本质和永久代类似,都是对JVM规范中方法区的实现. 元空间不在虚拟机中,而是使用本地内存. 用于元空间的JVM参数: -XX:MetaspaceSize=N 初始化Metaspace大小, ...
- jquery easyui datagrid 远程加载数据----把主键渲染为值遇到的问题及解决方案
起因:数据库中一些字段存的是代表具体值的数字,需要渲染为具体值 monggodb中的字典 mysql中存放的值为:expertin代表教练擅长的搏击技能 jquery easyui中的相关代码如下:用 ...
- CodeForces-939A
链接: https://vjudge.net/problem/CodeForces-939A 题意: As you could know there are no male planes nor fe ...
- ESP8266WiFiGeneric---通用库--事件和配置
ESP8266WiFiSTAClass .ESP8266WiFiScanClass .ESP8266WiFiAPClass 可以访问 ESP8266WiFiGenericClass的private和p ...
- 【NOIP2016提高A组模拟9.17】数格子
题目 分析 设表示每一行的状态,用一个4位的二进制来表示,当前这一行中的每一个位数对下一位有没有影响. 设\(f_{i,s}\)表示,做完了的i行,其状态为s,的方案数. 两个状态之间是否可以转移就留 ...
- [洛谷P3322] SDOI2015 排序
问题描述 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的 i(1<=i<=N),第i中操作为将序列 ...
- BZOJ 4897: [Thu Summer Camp2016]成绩单 动态规划
Description 期末考试结束了,班主任L老师要将成绩单分发到每位同学手中.L老师共有n份成绩单,按照编号从1到n的顺序叠 放在桌子上,其中编号为i的成绩单分数为w_i.成绩单是按照批次发放的. ...