总体步骤:init(); dfs1(1); dfs2(1,1); build(1,1,n); cl();dfs1 处理出size[sj](子节点数),son[sj](重儿子),dep[sj](深度),vi[sj](点权,把边权放在子节点上)。dfs2 处理出top[sj](所在链的链首,轻儿子从子节点重新开始,重儿子从父节点继承),id[sj](点出现的位置,用cnt标记),pos[sj](cnt位置的边连着哪一个点)。 id[x]=++cnt; pos[cnt]=x; if(son[x]) dfs2(son[x],y); for(int i=h[x];i!=-1;i=b[i].ne) if(b[i].v!=fa[x]&&b[i].v!=son[x]) dfs2(b[i].v,b[i].v);build if(z==y) mx[x]=vi[pos[y]]; 这个位置链的最大值等于这个位置的链所连点的权值。一般线段树方法建树。和线段树有关的数组要开四倍四倍四倍。。。cl 分change和query Change 改变的边编号要按输入数据检索,从线段树上找到这条边上作为子节点的点改变它的点权。 int now=(b1<<1)-1; int fr=b[now].u,to=b[now].v; if(fa[fr]==to) change(id[fr],b2,1,1,n); else change(id[to],b2,1,1,n); Query query函数即为一般的线段树查询,把不在一条链上的两个点向一条链上靠拢,时刻注意要求哪点在下层。如果本来就是点权,则id[x]不应该+1且id[x]==id[y]的情况是可行的。 int fx=top[x],fy=top[y],res=1<<31; while(fx^fy) { if(dep[fx]dep[y]) jh(x,y); if(id[x]