Language: Default
Tree
Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edges are numbered 1 through N − 1. Each edge is associated with a weight. Then you are to execute a series of instructions on the tree. The instructions can be one of the following forms:
Input The input contains multiple test cases. The first line of input contains an integer t (t ≤ 20), the number of test cases. Then follow the test cases. Each test case is preceded by an empty line. The first nonempty line of its contains N (N ≤ 10,000). The next N − 1 lines each contains three integers a, b and c, describing an edge connecting nodes a and b with weight c. The edges are numbered in the order they appear in the input. Below them are the instructions, each sticking to the specification above. A lines with the word “ Output For each “ Sample Input 1 3 1 2 1 2 3 2 QUERY 1 2 CHANGE 1 3 QUERY 1 2 DONE Sample Output 1 3 Source
POJ Monthly--2007.06.03, Lei, Tao
|
#include
#include
#include
#include
#define inf 0x3f3f3f3f
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
using namespace std;
const int maxn=50010;
struct Node{
int to,next;
}A[maxn<<1];
int MAX(int a,int b){
return a>b?a:b;
}
int edge,root,N,totw,d[maxn][3],fa[maxn];
int head[maxn],size[maxn],tree[maxn<<2];
int dep[maxn],pos[maxn],top[maxn],son[maxn];
void init(){
edge=totw=0;
memset(fa,0,sizeof(fa));
memset(dep,0,sizeof(dep));
memset(size,0,sizeof(size));
memset(head,-1,sizeof(head));
memset(tree,0,sizeof(tree));
}
void insert(int a,int b){
A[edge].to=b;
A[edge].next=head[a];
head[a]=edge++;
}
void dfs(int v){
size[v]=1;son[v]=0;
for(int k=head[v];k!=-1;k=A[k].next){
if(A[k].to!=fa[v]){
fa[A[k].to]=v;
dep[A[k].to]=dep[v]+1;
dfs(A[k].to);
size[v]+=size[A[k].to];
if(size[A[k].to]>size[son[v]])son[v]=A[k].to;
}
}
}
void buildtree(int v,int tp){
int k;pos[v]=++totw;top[v]=tp;
if(son[v])buildtree(son[v],top[v]);
for(int k=head[v];k!=-1;k=A[k].next){
if(A[k].to!=fa[v]&&A[k].to!=son[v]){
buildtree(A[k].to,A[k].to);
}
}
}
void change(int p,int x,int l,int r,int rt){
if(p>r||p>1;
change(p,x,lson);
change(p,x,rson);
tree[rt]=MAX(tree[rt<<1],tree[rt<<1|1]);
}
void Negate(int ll,int rr,int l,int r,int rt){
if(l==r){
tree[rt]*=-1;
return ;
}
int mid=(l+r)>>1;
if(ll<=mid)Negate(ll,rr,lson);
if(rr>mid)Negate(ll,rr,rson);
tree[rt]=MAX(tree[rt<<1],tree[rt<<1|1]);
}
void negate(int va,int vb){
int f1=top[va],f2=top[vb];
while(f1!=f2){
if(dep[f1]dep[vb])swap(va,vb);
Negate(pos[son[va]],pos[vb],1,totw,1);
}
int maxi(int ll,int rr,int l,int r,int rt){
if(ll>r||rr=ll&&r<=rr){
return tree[rt];
}
int mid=(l+r)>>1;
return MAX(maxi(ll,rr,lson),maxi(ll,rr,rson));
}
int find(int va,int vb){
int f1=top[va],f2=top[vb],tmp=-inf;
while(f1!=f2){
if(dep[f1]dep[vb])swap(va,vb);
return MAX(tmp,maxi(pos[son[va]],pos[vb],1,totw,1));
}
int main()
{
int n,m,q,t;
char str[10];
scanf("%d",&t);
while(t--){
scanf("%d",&n);
int a,b,c;init();
for(int i=1;idep[d[i][1]])swap(d[i][0],d[i][1]);
change(pos[d[i][1]],d[i][2],1,totw,1);
}
while(scanf("%s",str),str[0]!='D'){
if(str[0]=='Q'){
scanf("%d%d",&a,&b);
printf("%d\n",find(a,b));
}
else if(str[0]=='N'){
scanf("%d%d",&a,&b);
negate(a,b);
}
else if(str[0]=='C'){
scanf("%d%d",&a,&b);
change(pos[d[a][1]],b,1,totw,1);
}
}
}
return 0;
}