首页 > 代码库 > BZOJ1036: [ZJOI2008]树的统计Count
BZOJ1036: [ZJOI2008]树的统计Count
1036: [ZJOI2008]树的统计Count
Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4726 Solved: 1984
[Submit][Status]
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
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
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
1
2
2
10
6
5
6
5
16
HINT
Source
树的分治
做法:树链剖分
应该是最最最基础的树链剖分了,所谓树链剖分就是把树上的一段路径剖成很多条线段,想象一个“特殊的”dfs序,树上一条路径可以被分成dfs序上面一些线段。
由于剖出的线段最多只有log(n)个,所以树链剖分解题的复杂度是log(n) * 所用算法的复杂度,例如用线段树的话就是log(n) * log(n)
/*Author:wuhuajun*/#include <cstdio>#include <cstring>#include <cstdlib>#include <algorithm>#define lson l, mid, rt << 1#define rson mid+1, r, rt << 1 | 1using namespace std; typedef long long ll;typedef double dd;const int maxn=30010; int edge, n, fa[maxn], sz[maxn], son[maxn], dep[maxn], hash[maxn], top[maxn];int h[maxn], num, a[maxn], x, y, tx, ty, Q;char s[22]; struct Edge{ int to, ne;} e[maxn * 2]; struct Seg{ int maxx, sum; void clear() { maxx = -1000000000; sum = 0; }} seg[maxn << 2], ans; void close(){ exit(0);} void addedge(int x,int y){ e[edge].to = y; e[edge].ne = h[x]; h[x] = edge++;} void dfs(int k,int from){ sz[k] = 1; son[k] = 0; dep[k] = dep[from] + 1; for (int p=h[k];p!=-1;p=e[p].ne) { int to = e[p].to; if (from == to) continue; fa[to] = k; dfs(to, k); sz[k] += sz[to]; if (sz[to] > sz[son[k]]) son[k] = to; }} void build(int k,int from){ hash[k] = ++num; top[k] = from; if (son[k]) build(son[k], from); for (int p=h[k];p!=-1;p=e[p].ne) { int to = e[p].to; if (to != fa[k] && to != son[k]) build(to, to); }} //{{{Segment部分Seg update(Seg a,Seg b){ Seg c; c.sum = a.sum + b.sum; c.maxx = max(a.maxx, b.maxx); return c;} void pushup(int rt){ seg[rt].maxx = max(seg[rt<<1].maxx, seg[rt<<1|1].maxx); seg[rt].sum = seg[rt<<1].sum + seg[rt<<1|1].sum;} void change(int L,int R,int val,int l,int r,int rt){ if (L <= l && r <= R) { seg[rt].sum = seg[rt].maxx = val; return; } int mid = (l + r) >> 1; if (L <= mid) change(L,R,val,lson); if (mid + 1 <= R) change(L,R,val,rson); pushup(rt);} Seg query(int L,int R,int l,int r,int rt){ if (L <= l && r <= R) { return seg[rt]; } int mid = (l + r) >> 1; Seg ans, a, b; ans.clear(); a.clear(); b.clear(); if (L <= mid) a = query(L,R,lson); if (mid + 1 <= R) b = query(L,R,rson); ans.sum = a.sum + b.sum; ans.maxx = max(a.maxx, b.maxx); return ans;}//}}} Seg get_ans(){ tx = top[x]; ty = top[y]; ans.clear(); while (tx != ty) { if (dep[tx] < dep[ty]) { swap(tx, ty); swap(x, y); } ans = update(ans, query(hash[tx], hash[x], 1, n, 1)); x = fa[tx]; tx = top[x]; } if (dep[x] > dep[y]) swap(x, y); return update(ans, query(hash[x], hash[y], 1, n, 1));} void init(){ scanf("%d",&n); memset(h,-1,sizeof(h)); for (int i=1;i<=n-1;i++) { scanf("%d %d",&x, &y); addedge(x, y); addedge(y, x); } for (int i=1;i<=n;i++) scanf("%d", &a[i]); dfs(1, 0); build(1, 1); /* for (int i=1;i<=n;i++) { printf("i:%d top:%d hash:%d\n",i, top[i], hash[i]); } */ for (int i=1;i<=n;i++) change(hash[i], hash[i], a[i], 1, n, 1); scanf("%d",&Q); while (Q--) { scanf("%s %d %d",s,&x,&y); if (s[0] == ‘C‘) { change(hash[x], hash[x], y, 1, n, 1); continue; } ans = get_ans(); if (s[1] == ‘M‘) printf("%d\n", ans.maxx); else printf("%d\n", ans.sum); }} int main (){ init(); close(); return 0;}
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。