一 棵樹上有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自己php
一 棵樹上有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自己php
輸 入的第一行爲一個整數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之間。算法
對於每一個「QMAX」或者「QSUM」的操做,每行輸出一個整數表示要求輸出的結果。ui
樹的分治spa
作法:樹鏈剖分blog
應該是最最最基礎的樹鏈剖分了,所謂樹鏈剖分就是把樹上的一段路徑剖成不少條線段,想象一個「特殊的」dfs序,樹上一條路徑能夠被分紅dfs序上面一些線段。ip
因爲剖出的線段最多隻有log(n)個,因此樹鏈剖分解題的複雜度是log(n) * 所用算法的複雜度,例如用線段樹的話就是log(n) * log(n)get
/* Author:wuhuajun */ #include <cstdio> #include <cstring> #include <cstdlib> #include <algorithm> #define lson l, mid, rt << 1 #define rson mid+1, r, rt << 1 | 1 using 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; }