BZOJ 1036 树的统计
Description
一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 III. 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
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
1
2
2
10
6
5
6
5
16
HINT
Source
#include<cstdio>
#include<cstdlib>
#include<cstring>
using namespace std;
#define maxn 30010
#define root 1
int n,next[*maxn],side[maxn],toit[*maxn],key[maxn],tot,son[maxn],father[maxn];
int edge[maxn],pos[maxn],top[maxn],dep[maxn],heavy[maxn],ord[maxn];
bool in[maxn];
struct node
{
int l,r;
int best,sum;
int lc,rc;
}seg[*maxn]; inline int big(int a,int b){if (a > b)return a;return b;}
inline void add(int,int);
inline void dfs(int,int);
inline void link(int,int);
inline void build(int,int);
inline void change(int,int);
inline void find_best(int,int);
inline void find_sum(int,int);
inline int seg_sum(int,int,int);
inline int seg_best(int,int,int); int main()
{
freopen("1036.in","r",stdin);
freopen("1036.out","w",stdout);
scanf("%d",&n);
int i,a,b;
for (i = ;i<n;i++)
{
scanf("%d %d",&a,&b);
add(a,b); add(b,a);
}
for (i = ;i<=n;i++)
scanf("%d",key+i);
tot = ;
dfs(root,);
tot = ;
memset(in,,sizeof(in));
link(root,root);
tot = ;
build(,n);
int T;
scanf("%d\n",&T);
char sign[];
while (T)
{
T--;
scanf("%s %d %d\n",sign,&a,&b);
if (sign[] == 'Q')
{
if (sign[] == 'M')
find_best(a,b);
if (sign[] == 'S')
find_sum(a,b);
}
else
key[a] = b,change(a,); }
return ;
} inline void add(int a,int b)
{
tot++;
toit[tot] = b;
next[tot] = side[a];
side[a] = tot;
} inline void dfs(int a,int deep)
{
in[a] = true; dep[a] = deep;
int u = side[a],v;
son[a] = ;
while (u != )
{
v = toit[u];
if (!in[v])
{
father[v] = a;
edge[++tot] = v;
dfs(v,deep+);
if (son[heavy[a]] < son[v])
heavy[a] = v;
son[a] += son[v];
}
u = next[u];
}
} inline void link(int a,int high)
{
top[a] = high;
pos[a] = ++tot;
ord[tot] = a;
in[a] = true;
if (heavy[a] != )
link(heavy[a],high);
else return;
int u = side[a],v;
while (u != )
{
v = toit[u];
if (!in[v])
link(v,v);
u = next[u];
}
} inline void build(int l,int r)
{
seg[tot].l = l;
seg[tot].r = r;
if (l == r)
{
seg[tot].best = key[ord[l]];
seg[tot].sum = key[ord[l]];
return;
}
int mid = ((l+r)>>);
int k = tot;
seg[k].lc = ++tot;
build(l,mid);
seg[k].rc = ++tot;
build(mid+,r);
seg[k].best = big(seg[seg[k].lc].best,seg[seg[k].rc].best);
seg[k].sum = seg[seg[k].lc].sum+seg[seg[k].rc].sum;
} inline void change(int a,int now)
{
if (seg[now].l == seg[now].r)
{
seg[now].best = key[a];
seg[now].sum = key[a];
return;
}
if (seg[seg[now].lc].l<=pos[a]&&seg[seg[now].lc].r>=pos[a])
change(a,seg[now].lc);
else change(a,seg[now].rc);
seg[now].best = big(seg[seg[now].lc].best,seg[seg[now].rc].best);
seg[now].sum = seg[seg[now].lc].sum + seg[seg[now].rc].sum;
} inline void find_sum(int a,int b)
{
int sum = ;
while (top[a] != top[b])
{
if (dep[top[a]] >= dep[top[b]])
{
sum += seg_sum(pos[top[a]],pos[a],);
a = father[top[a]];
}
else
{
sum += seg_sum(pos[top[b]],pos[b],);
b = father[top[b]];
}
}
if (pos[a] <= pos[b])
sum += seg_sum(pos[a],pos[b],);
else
sum += seg_sum(pos[b],pos[a],);
printf("%d\n",sum);
} inline void find_best(int a,int b)
{
int best = -;
while (top[a] != top[b])
{
if (dep[top[a]] >= dep[top[b]])
{
best = big(best,seg_best(pos[top[a]],pos[a],));
a = father[top[a]];
}
else
{
best = big(best,seg_best(pos[top[b]],pos[b],));
b = father[top[b]];
}
}
if (pos[a] <= pos[b])
best = big(best,seg_best(pos[a],pos[b],));
else
best = big(best,seg_best(pos[b],pos[a],));
printf("%d\n",best);
} inline int seg_best(int l,int r,int now)
{
if (seg[now].l >=l && seg[now].r <=r)
return seg[now].best;
int ret =-;
int t;
t = seg[now].lc;
if (seg[t].l<=l&&seg[t].r>=l)
ret = big(ret,seg_best(l,r,t));
else if (seg[t].l>=l&&seg[t].l<=r)
ret = big(ret,seg_best(l,r,t));
else if (seg[t].r>=l&&seg[t].r<=r)
ret = big(ret,seg_best(l,r,t));
t = seg[now].rc;
if (seg[t].l<=l&&seg[t].r>=l)
ret = big(ret,seg_best(l,r,t));
else if (seg[t].l>=l&&seg[t].l<=r)
ret = big(ret,seg_best(l,r,t));
else if (seg[t].r>=l&&seg[t].r<=r)
ret = big(ret,seg_best(l,r,t));
return ret;
} inline int seg_sum(int l,int r,int now)
{
if (seg[now].l >=l && seg[now].r <=r)
return seg[now].sum;
int ret = ;
int t;
t = seg[now].lc;
if (seg[t].l<=l&&seg[t].r>=l)
ret += seg_sum(l,r,t);
else if (seg[t].l>=l&&seg[t].l<=r)
ret += seg_sum(l,r,t);
else if (seg[t].r>=l&&seg[t].r<=r)
ret += seg_sum(l,r,t);
t = seg[now].rc;
if (seg[t].l<=l&&seg[t].r>=l)
ret += seg_sum(l,r,t);
else if (seg[t].l>=l&&seg[t].l<=r)
ret += seg_sum(l,r,t);
else if (seg[t].r>=l&&seg[t].r<=r)
ret += seg_sum(l,r,t);
return ret;
}
BZOJ 1036 树的统计的更多相关文章
- BZOJ 1036 树的统计-树链剖分
[ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 12904 Solved: 5191[Submit][Status ...
- Codevs 2460 == BZOJ 1036 树的统计
2460 树的统计 2008年省队选拔赛浙江 时间限制: 2 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 一棵树上有n个节点,编号分别为1 ...
- BZOJ 1036 树的统计Count 树链剖分模板题
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...
- [置顶] bzoj 1036 树的统计Count 点权值模板
树链剖分 点权型可做模板,链路剖分的思想把点hash到线段树的上,然后可通过n*(log(n)*log(n))的复杂度在树上操作,在线段树上能操作的在链路上都能操作. #include<cstd ...
- BZOJ 1036 树的统计(树链剖分)
点权树链剖分模板题. # include <cstdio> # include <cstring> # include <cstdlib> # include &l ...
- bzoj 1036 树的统计Count
题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u ...
- BZOJ - 1036 树的统计Count (LCT)
LCT试炼题(代码量居然完爆树剖?) #include<bits/stdc++.h> using namespace std; ,inf=0x3f3f3f3f; ],flp[N],n,m, ...
- Bzoj 1036 树的统计 分类: ACM TYPE 2014-12-29 18:55 72人阅读 评论(0) 收藏
Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. Q ...
- BZOJ 1036 树的统计 | 树链剖分模板题
又做了一遍--去掉读入优化只有八十行~ #include <cstdio> #include <cstring> #include <algorithm> usin ...
随机推荐
- jsp页面写入中文到mysql时出现了乱码(转)
今天自己在用jsp把中文写入mysql的时候出现乱码,从数据库中读取出来的时候也显示为“??”,感觉应该出现了编码转换过程中的字符信息丢失.然后在mysql中直接执行该命令,发现中文是正常的,所有认为 ...
- 浅谈UML中类之间的五种关系及其在代码中的表现形式
本文转载:http://www.cnblogs.com/DebugLZQ/archive/2013/05/13/3066715.html 什么是类? 将某类东西归纳在一起,可以成为一个类. 类有很多种 ...
- javascript,jquery(闭包概念)(转)
偶尔听人说javascript闭包,让我联想起以前学编译原理和数字逻辑里讲的闭包,以前上课讲的闭包很难懂,而且含有递归的意思在里面,现在不想再查看里面的闭包概念. 但javascript我是经常要用, ...
- UML进行Linux内核调试
http://www.lenky.info/ http://blog.csdn.net/ztz0223/article/details/7874759 http://user-mode-linux.s ...
- 基于HTML5的SLG游戏开发(一):搭建开发环境(1)
开发环境: 1.操作系统:MacOS 10.8.5 2.本地web服务器:Apache 2.2.24 (Window环境推荐使用WampServer) 3.编码工具:WebStrom 7.0 4.调试 ...
- UIPickView的简单介绍
UIPickView的简单介绍 设置UIPickView的时候,我们主要需要设置一下下面的两个属性 UIPickerView *pickView1; pickView1 = [[UIPickerVie ...
- 解决github提交commit,contributions不统计显示绿色的问题
最近使用GitHub Desktop时,发现自己好多次的commits都没有被记录在Contributions中,但是点开项目详情里面可以看到自己的commit确实上传成功了,所以就忧伤了,为什么没有 ...
- 阿里大于验证码发送 (ThinkPhp框架)
1.登录平台 阿里大于2.登陆之后我们可以看到资费,使用场景等,在进入正题之前我们需要一些准备工作,首先我们先了解下短信的请求参数,在这里我们需要注意的是sms_param这个参数,在接下来我们申请短 ...
- 10.24 noip模拟试题
尼玛pdf依旧不会粘23333 /* 每段合并到总的里面 假设总的有X个 这一段有Y个 一共有X+1个空 那么就有 C(X+1,1)+C(X+1,2)+C(X+1,3)+...+C(X+1,Y) 这样 ...
- C#总结项目《影院售票系统》编写总结一
C#学习经历从基本语法结构到窗体再到面向对象终于走完了.NET初级程序员的道路,做为品德优良好学生更不能落下课程的总结项目-某某鸟<影院售票系统>.用了大概一天半的时间做完这个练手项目,先 ...