题目链接: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 <cstring>
using namespace std;
const int M = 5e4 + 10;
struct Edge {
int v , next , w;
}edge[M << 1];
int head[M] , e;//链式前向星存边
int top[M];//重链的根节点
int fa[M];//fa[i]表示i的父节点
int deep[M];//节点的深度
int num[M];//num[i]表示i节点子树一共有多少节点
int p[M];//p[i]表示i点在线段树中的位置
int fp[M];//fp[i]表示线段树中位置为i的节点是什么
int son[M];//sum[u]表示u点的重儿子
int pos;
void init() {
memset(head , -1 , sizeof(head));
memset(son , -1 , sizeof(son));
e = 0;
pos = 1;
}//初始化
void add(int u , int v) {
edge[e].v = v;
edge[e].next = head[u];
head[u] = e++;
}//加边
void dfs1(int u , int pre , int d) {
deep[u] = d;
fa[u] = pre;
num[u] = 1;
for(int i = head[u] ; i != -1 ; i = edge[i].next) {
int v = edge[i].v;
if(v != pre) {
dfs1(v , u , d + 1);
num[u] += num[v];
if(son[u] == -1 || num[v] > num[son[u]]) {
son[u] = v;
}
}
}
}//寻找重点
void getpos(int u , int sp) {
top[u] = sp;
p[u] = pos++;
fp[pos - 1] = u;
if(son[u] == -1)
return ;
getpos(son[u] , sp);
for(int i = head[u] ; i != -1 ; i = edge[i].next) {
int v = edge[i].v;
if(v != son[u] && v != fa[u]) {
getpos(v , v);
}
}
}//构建重链
struct TnT {
int l , r , sum , lazy;
}T[M << 2];
int a[M];
void build(int l , int r , int i) {
int mid = (l + r) >> 1;
T[i].l = l , T[i].r = r , T[i].lazy = 0 , T[i].sum = 0;
if(T[i].l == T[i].r) {
T[i].sum = a[fp[l]];
return ;
}
build(l , mid , i << 1);
build(mid + 1 , r , (i << 1) | 1);
T[i].sum = T[i << 1].sum + T[(i << 1) | 1].sum;
}
void pushdown(int i) {
if(T[i].lazy) {
T[i << 1].sum += T[i].lazy * (T[i << 1].r - T[i << 1].l + 1);
T[(i << 1) | 1].sum += T[i].lazy * (T[(i << 1) | 1].r - T[(i << 1) | 1].l + 1);
T[i << 1].lazy += T[i].lazy;
T[(i << 1) | 1].lazy += T[i].lazy;
T[i].lazy = 0;
}
}
void updata(int l , int r , int ad , int i) {
int mid = (T[i].l + T[i].r) >> 1;
if(T[i].l == l && T[i].r == r) {
T[i].sum += ad * (r - l + 1);
T[i].lazy += ad;
return ;
}
pushdown(i);
if(mid < l) {
updata(l , r , ad , (i << 1) | 1);
}
else if(mid >= r) {
updata(l , r , ad , i << 1);
}
else {
updata(l , mid , ad , i << 1) , updata(mid + 1 , r , ad , (i << 1) | 1);
}
T[i].sum = T[i << 1].sum + T[(i << 1) | 1].sum;
}
int query(int i , int pos) {
int mid = (T[i].l + T[i].r) >> 1;
if(T[i].l == pos && T[i].r == pos) {
return T[i].sum;
}
pushdown(i);
T[i].sum = T[i << 1].sum + T[(i << 1) | 1].sum;
if(mid < pos) {
return query((i << 1) | 1 , pos);
}
else {
return query(i << 1 , pos);
}
}
void change(int u , int v , int ad) {
int f1 = top[u] , f2 = top[v];
while(f1 != f2) {
if(deep[f1] < deep[f2]) {
swap(f1 , f2);
swap(u , v);
}
updata(p[f1] , p[u] , ad , 1);
u = fa[f1] , f1 = top[u];
}
if(deep[u] < deep[v])
swap(u , v);
updata(p[v] , p[u] , ad , 1);
}//树链剖分特别的更新方法,类似于lca从各子链上查询直到到了同一条重链上自行理解一下。
int main() {
int n , m , q , x , y , z;
while(scanf("%d%d%d" , &n , &m , &q) != EOF) {
init();
for(int i = 1 ; i <= n ; i++) {
scanf("%d" , &a[i]);
}
for(int i = 0 ; i < m ; i++) {
scanf("%d%d" , &x , &y);
add(x , y);
add(y , x);
}
dfs1(1 , 0 , 0);
getpos(1 , 1);
char cp[5];
build(0 , pos + 1 , 1);
while(q--) {
scanf("%s" , cp);
if(cp[0] == 'I') {
scanf("%d%d%d" , &x , &y , &z);
change(x , y , z);
}
if(cp[0] == 'D') {
scanf("%d%d%d" , &x , &y , &z);
change(x , y , -z);
}
if(cp[0] == 'Q') {
scanf("%d" , &x);
printf("%d\n" , query(1 , p[x]));
}
}
}
return 0;
}

hdu 3966 Aragorn's Story(树链剖分+区间修改+单点查询)的更多相关文章

  1. HDU 3966 Aragorn's Story 树链剖分+树状数组 或 树链剖分+线段树

    HDU 3966 Aragorn's Story 先把树剖成链,然后用树状数组维护: 讲真,研究了好久,还是没明白 树状数组这样实现"区间更新+单点查询"的原理... 神奇... ...

  2. Hdu 3966 Aragorn's Story (树链剖分 + 线段树区间更新)

    题目链接: Hdu 3966 Aragorn's Story 题目描述: 给出一个树,每个节点都有一个权值,有三种操作: 1:( I, i, j, x ) 从i到j的路径上经过的节点全部都加上x: 2 ...

  3. HDU 3966 Aragorn's Story(树链剖分)(线段树区间修改)

    Aragorn's Story Time Limit: 10000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  4. HDU 3966 Aragorn's Story 树链剖分+BIT区间修改/单点询问

    Aragorn's Story Description Our protagonist is the handsome human prince Aragorn comes from The Lord ...

  5. HDU 3966 Aragorn's Story (树链剖分入门题)

    树上路径区间更新,单点查询. 线段树和树状数组都可以用于本题的维护. 线段树: #include<cstdio> #include<iostream> #include< ...

  6. HDU 3966 Aragorn's Story 树链剖分

    Link: http://acm.hdu.edu.cn/showproblem.php?pid=3966 这题注意要手动扩栈. 这题我交g++无限RE,即使手动扩栈了,但交C++就过了. #pragm ...

  7. hdu 3966 Aragorn's Story : 树链剖分 O(nlogn)建树 O((logn)²)修改与查询

    /** problem: http://acm.hdu.edu.cn/showproblem.php?pid=3966 裸板 **/ #include<stdio.h> #include& ...

  8. hdu 3966 Aragorn's Story 树链剖分 按点

    Aragorn's Story Time Limit: 10000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  9. 【树状数组区间修改单点查询+分组】HDU 4267 A Simple Problem with Integers

    http://acm.hdu.edu.cn/showproblem.php?pid=4267 [思路] 树状数组的区间修改:在区间[a, b]内更新+x就在a的位置+x. 然后在b+1的位置-x 树状 ...

随机推荐

  1. Github上fork的项目如何merge原Git项目

    问题场景 小明在Github上fork了一个大佬的项目,并clone到本地开发一段时间,再提交merge request到原Git项目,过了段时间,原作者联系小明,扔给他下面这幅截图并告知合并处理冲突 ...

  2. ipad pro 为什么不行

    TalkingData公布的数据显示,iPad Pro在中国发行首月的销量仅为49 300台,而此前iPad Air 2发行首月后销量曾高达55.7万台.那么到底是什么原因,让这个被寄予厚望的iPad ...

  3. C#下载文件,Stream 和 byte[] 之间的转换

    stream byte 等各类转换 http://www.cnblogs.com/warioland/archive/2012/03/06/2381355.html using (System.Net ...

  4. IE浏览器主页被篡改为2345,针对一般解决办法无法解决的情况

    1.注册表修改 按微软键+R  输入regedit 弹出注册表.HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Internet Explorer\MAIN 将右侧的Sta ...

  5. Powered by .NET Core 进展:第5次发布尝试(Windows部署)

    (图注:Windows 自带的性能监控,红色表示 CPU 占用,绿色表示 QPS) 今天中午 12:30 左右,我们进行了 .NET Core 博客站点的第5次发布(页脚会显示"Powere ...

  6. 7.15 迭代器 for循环的本质 生成器

    迭代器 迭代:更新换代的过程,每次的迭代都必须基于上一次的结果 迭代器:迭代取值的工具 作用 迭代器提供了一种不依赖于索引取值的方式 根据以上对于迭代的描述,如果只是简单的重复,不算迭代,如下: n ...

  7. Java 从入门到进阶之路(一)

    写在前面:从本片文章开始,将记录自己学习 Java 的点滴路程,目标定的并不是让自己成为一个 Java 高手,而是让自己多掌握一门语言,使自己的知识面更广一些,在学习 Java 的过程中如有不对的地方 ...

  8. 十分钟搭建微服务框架(SpringBoot +Dubbo+Docker+Jenkins源码)

    本文将以原理+实战的方式,首先对“微服务”相关的概念进行知识点扫盲,然后开始手把手教你搭建这一整套的微服务系统. 这套微服务框架能干啥? 这套系统搭建完之后,那可就厉害了: 微服务架构 你的整个应用程 ...

  9. idea 2019安装完(打不开&&启动不了)问题解决(最全解决方法)

    今天从网盘把idea下载下来后一路安装,准备 设置的时候不管怎么打开 他都无动于衷没办法,卸了安,安了卸,反复折腾了 好几遍 它都无动于衷.于是开始在百度上找答案看了 好几个 方法一遍一遍试还是不行, ...

  10. 10G的变态SQL文件,如何快速打开编辑?

    工作中,偶尔需要编辑一些大文件,比如 log 文件,后者一些变态的 SQL,此时用平常的编辑器就会显得力不从心,要么直接打不开,要么打开后卡得要死. 本文就给大家推荐几款可以操作大文件的编辑器,准备好 ...