描述


http://www.lydsy.com/JudgeOnline/problem.php?id=1036

给出一棵树以及各点的权值,对数进行如下三种操作:

1.改变某一节点u的值为t;

2.求节点u到节点v之间(包括u与v)的最大值;

3.求节点u到节点v之间(包括u与v)的和.

1036: [ZJOI2008]树的统计Count

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 12002  Solved: 4864
[Submit][Status][Discuss]

Description

  一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。我们将以下面的形式来要求你对这棵树完成
一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 I
II. QSUM u v: 询问从点u到点v的路径上的节点的权值和 注意:从点u到点v的路径上的节点包括u和v本身

Input

  输入的第一行为一个整数n,表示节点的个数。接下来n – 1行,每行2个整数a和b,表示节点a和节点b之间有
一条边相连。接下来n行,每行一个整数,第i行的整数wi表示节点i的权值。接下来1行,为一个整数q,表示操作
的总数。接下来q行,每行一个操作,以“CHANGE u t”或者“QMAX u v”或者“QSUM u v”的形式给出。
对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。

Output

  对于每个“QMAX”或者“QSUM”的操作,每行输出一个整数表示要求输出的结果。

Sample Input

4
1 2
2 3
4 1
4 2 1 3
12
QMAX 3 4
QMAX 3 3
QMAX 3 2
QMAX 2 3
QSUM 3 4
QSUM 2 1
CHANGE 1 5
QMAX 3 4
CHANGE 3 6
QMAX 3 4
QMAX 2 4
QSUM 3 4

Sample Output

4
1
2
2
10
6
5
6
5
16

HINT

Source

分析


树链剖分模板题,但貌似LCA写得太原始了= =,并不会其他方法.直接裸的原始LCA好像可以过两个点...

注意:

1.对u,v进行操作时是判断tib[u]与tib[v]的大小,不然上线段树会炸.(tib[top[u]]<=tib[u]恒成立)

2.在LCA时别少写了dep[top[u]]==dep[top[v]]的情况.

 #include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm>
#define read(a) a=getnum()
#define lson (2*k)
#define rson (2*k+1)
#define mid ((a[k].l+a[k].r)>>1) using namespace std; const int maxn=+,INF=+; struct node
{
int l,r,k,x,m;
}; int n,q;
int label;
int w[maxn];
int fa[maxn];
int dep[maxn];
int size[maxn];
int son[maxn];
int top[maxn];
int tib[maxn];
int rev_tib[maxn];
bool vis1[maxn];
bool vis2[maxn];
vector <int> g1[maxn];
vector <int> g2[maxn];
node a[*maxn]; inline int getnum()
{
int r=,k=;
char c;
for(c=getchar();c<''||c>'';c=getchar()) if(c=='-') k=-;
for(;c>=''&&c<='';c=getchar()) r=r*+c-'';
return r*k;
} void build_tree(int l,int r,int k)
{
a[k].l=l; a[k].r=r;
if(l==r)
{
a[k].x=a[k].m=w[rev_tib[l]];
return;
}
build_tree(l,mid,lson);
build_tree(mid+,r,rson);
a[k].x=a[lson].x+a[rson].x;
a[k].m=max(a[lson].m,a[rson].m);
} void update(int x,int t,int k)
{
if(a[k].l==a[k].r&&a[k].l==x)
{
a[k].x=a[k].m=t;
return;
}
if(x<=mid) update(x,t,lson);
else update(x,t,rson);
a[k].x=a[lson].x+a[rson].x;
a[k].m=max(a[lson].m,a[rson].m);
} int get_max(int l,int r,int k)
{
if(l==a[k].l&&r==a[k].r)
{
return a[k].m;
}
if(r<=mid) return get_max(l,r,lson);
else if(l>mid) return get_max(l,r,rson);
else return max(get_max(l,mid,lson),get_max(mid+,r,rson));
} int get_sum(int l,int r,int k)
{
if(l==a[k].l&&r==a[k].r)
{
return a[k].x;
}
if(r<=mid) return get_sum(l,r,lson);
else if(l>mid) return get_sum(l,r,rson);
else return get_sum(l,mid,lson)+get_sum(mid+,r,rson);
} void find_h_e(int u,int father,int depth)
{
vis1[u]=true;
fa[u]=father;
dep[u]=depth;
size[u]=;
son[u]=;
int max_size=;
for(int i=;i<g1[u].size();i++)
{
int v=g1[u][i];
if(vis1[v]) continue;
g2[u].push_back(v);
find_h_e(v,u,depth+);
size[u]+=size[v];
if(size[v]>max_size)
{
max_size=size[v];
son[u]=v;
}
}
} void conect_h_e(int u,int ancestor)
{
vis2[u]=true;
top[u]=ancestor;
tib[u]=++label;
rev_tib[label]=u;
if(son[u])
{
conect_h_e(son[u],ancestor);
}
for(int i=;i<g2[u].size();i++)
{
int v=g2[u][i];
if(vis2[v]) continue;
conect_h_e(v,v);
}
} void Change(int u,int t)
{
update(tib[u],t,);
} int Q_max(int u,int v)
{
int max_now=-INF;
while(top[u]!=top[v])
{
if(dep[top[u]]==dep[top[v]])
{
max_now=max(max_now,get_max(tib[top[u]],tib[u],));
u=fa[top[u]];
continue;
}
while(dep[top[u]]>dep[top[v]])
{
max_now=max(max_now,get_max(tib[top[u]],tib[u],));
u=fa[top[u]];
}
while(dep[top[u]]<dep[top[v]])
{
max_now=max(max_now,get_max(tib[top[v]],tib[v],));
v=fa[top[v]];
}
}
max_now=max(max_now,get_max(min(tib[v],tib[u]),max(tib[u],tib[v]),));
return max_now;
} int Q_sum(int u,int v)
{
int sum_now=;
while(top[u]!=top[v])
{
if(dep[top[u]]==dep[top[v]])
{
sum_now+=get_sum(tib[top[u]],tib[u],);
u=fa[top[u]];
continue;
}
while(dep[top[u]]>dep[top[v]])
{
sum_now+=get_sum(tib[top[u]],tib[u],);
u=fa[top[u]];
}
while(dep[top[u]]<dep[top[v]])
{
sum_now+=get_sum(tib[top[v]],tib[v],);
v=fa[top[v]];
}
}
sum_now+=get_sum(min(tib[v],tib[u]),max(tib[u],tib[v]),);
return sum_now;
} void init()
{
read(n);
for(int i=;i<n;i++)
{
int u,v;
read(u); read(v);
g1[u].push_back(v);
g1[v].push_back(u);
}
for(int i=;i<=n;i++)
{
read(w[i]);
}
} int main()
{
init();
read(q);
find_h_e(,,);
conect_h_e(,);
build_tree(,n,);
for(int i=;i<=q;i++)
{
char c;
c=getchar();
if(c=='C')
{
int u,t;
read(u); read(t);
Change(u,t);
}
else
{
c=getchar();
if(c=='M')
{
int u,v;
read(u); read(v);
printf("%d\n",Q_max(u,v));
}
else
{
int u,v;
read(u); read(v);
printf("%d\n",Q_sum(u,v));
}
}
}
return ;
}

BZOJ_1036_[ZJOI2008]_树的统计Conut_(树链剖分)的更多相关文章

  1. (持续更新)虚树,KD-Tree,长链剖分,后缀数组,后缀自动机

    真的就是讲课两天,吸收一个月呢! \(1.\)虚树 \(2.\)KD-Tree \(3.\)长链剖分 \(4.\)后缀数组 后缀数组 \(5.\)后缀自动机 后缀自动机

  2. BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】

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

  3. 【BZOJ1036】[ZJOI2008]树的统计Count 树链剖分

    [BZOJ1036][ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. ...

  4. Bzoj 1036: [ZJOI2008]树的统计Count 树链剖分,LCT

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

  5. BZOJ 1036: [ZJOI2008]树的统计Count( 树链剖分 )

    树链剖分... 不知道为什么跑这么慢 = = 调了一节课啊跪.. ------------------------------------------------------------------- ...

  6. bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题

    [ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...

  7. P2590 [ZJOI2008]树的统计(树链剖分)

    P2590 [ZJOI2008]树的统计 虽然是入门树剖模板 但是我终于1A了(大哭) 懒得写啥了(逃 #include<iostream> #include<cstdio> ...

  8. bzoj 1036: [ZJOI2008]树的统计Count 树链剖分+线段树

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

  9. BZOJ 1036: [ZJOI2008]树的统计Count (树链剖分模板题)

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

随机推荐

  1. Ext.Net学习笔记04:Ext.Net布局

    ExtJS中的布局功能很强大,常用的布局有border.accordion.fit.hbox.vbox等,Ext.Net除了将这些布局进行封装以外,更是对border进行了一些非常实用的改进,让我们来 ...

  2. Javascript模仿C语言的链表实现(增删改查),并且使用控制台输入输出

    Js新手最近在研究Js数据结构,刚好看到链表实现这一块儿,觉得有些资料和自己理解的有冲突,于是借着自己以前一点点C语言的基础,用Javascript模仿了C的链表实现,并且用了process.stdi ...

  3. 10.10_魔兽账号,OSC代码托管演示,研究SQL别忘记了,git

    (1)juedui8456juedui456chixin0769魔兽世界账号112288 (2)EasyXls.开源中国推出 PaaS@OSC 代码演示和运行平台.git.oschina.coding ...

  4. 09.13日记(2014年9月13日00:18:26)英语,bootstrap,阮一峰,

    我们这里只推荐一本语法书:台湾的旋元佑老师写的<文法俱乐部>(简体版名为<语法俱乐部>).这本书因为出版社倒闭而绝版,淘宝可以买到影印的版本. (1)学英语:奶爸的英语教室 资 ...

  5. linux采用模块方法,添加一个新的设备

    该文转载自:http://rangercyh.blog.51cto.com/1444712/521244 系统调用是操作系统内核和应用程序之间的接口,而设备驱动程序是操作系统内核和机器硬件之间的接口. ...

  6. java中移位运算符:<<、>>和>>>之间的比较

    一.说明 <<:运算符将二进制位进行左移操作 >>:运算符将二进制位进行右移操作 >>>:运算符将用0填空高位 二.举例 /** * *----------c ...

  7. 『奇葩问题集锦』Ruby 切换淘宝源报错WARNING: Error fetching data: SSL_connect returned=1 errno=0 state=SSLv3 read s erver certificate B: certificate verify failed

    ===>首先需要使用https<===https://ruby.taobao.org/ 第一步 下载http://pan.baidu.com/s/1kU0rxtH 复制到ruby安装的根目 ...

  8. mysql自动备份数据库

    可以选择设置需要备份的库,自动备份压缩,自动删除 7 天前的备份,需要使用 crontab 定时执行. #!/bin/bash # 要备份的数据库名,多个数据库用空格分开 databases=(db1 ...

  9. STM32系统时钟

    一.时钟树 STM32有4个时钟源: 1)HSE(高速外部时钟源) 外部晶振作为时钟源,范围为4~16MHz,常取为8MHz 2)HSI(高速内部时钟源) 由内部RC振荡器产生,频率为8MHz,但不稳 ...

  10. 4069: [Apio2015]巴厘岛的雕塑

    Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄 ...