Santiego @ 2019-10-31 11:12:46
已跪调一上午,请各位大佬看一眼,本蒟蒻不胜感激!
线段树合并做法,rot1[]
维护上行路径的差分权值线段树,rot2[]
维护下行路径
#include <cstdio>
#include <algorithm>
#define MAXN 300003
using namespace std;
inline int read(){
char ch=getchar();int s=0;bool w=0;
while((ch<'0'||ch>'9')&&(ch!='-')) ch=getchar();
if(ch=='-'){w=1;ch=getchar();}
while(ch>='0'&&ch<='9') s=s*10+(ch^'0'),ch=getchar();
if(w) return -s;
return s;
}
int head[MAXN],nxt[MAXN*2],vv[MAXN*2],tot;
inline void add_egde(int u, int v){
vv[++tot]=v;
nxt[tot]=head[u];
head[u]=tot;
}
int dep[MAXN],mxs[MAXN],sz[MAXN],f[MAXN];
void dfs(int u, int fa){
sz[u]=1;
dep[u]=dep[fa]+1;
f[u]=fa;
int mxsz=-1;
for(int i=head[u];i;i=nxt[i]){
int v=vv[i];
if(v==fa) continue;
dfs(v, u);
sz[u]+=sz[v];
if(sz[v]>mxsz){
mxsz=sz[v];
mxs[u]=v;
}
}
}
int topf[MAXN];
void dfs2(int u, int top){
topf[u]=top;
if(mxs[u]==0) return;
dfs2(mxs[u], top);
for(int i=head[u];i;i=nxt[i]){
int v=vv[i];
if(v==f[u]||v==mxs[u]) continue;
dfs2(v, v);
}
}
int lca(int a,int b){
while(topf[a]!=topf[b]){
if(dep[a]<dep[b]) swap(a,b);
a=f[topf[a]];
}
if(dep[a]<dep[b]) return a;
else return b;
}
int cnt;
#define MAXM MAXN*2*18*2
int tre[MAXM],sl[MAXM],sr[MAXM];
void change(int &x, int l, int r, int pos, int val){
if(x==0) x=++cnt;
if(l==r){
tre[x]+=val;
return;
}
int mid=(l+r)>>1;
if(pos<=mid) change(sl[x], l, mid, pos, val);
else change(sr[x], mid+1, r, pos, val);
}
int query(int x, int l, int r, int pos){
if(x==0) return 0;
if(l==r) return tre[x];
int mid=(l+r)>>1;
if(pos<=mid) return query(sl[x], l, mid, pos);
else return query(sr[x], mid+1, r, pos);
}
int merge(int a, int b, int l, int r){
if(a==0||b==0) return a+b;
if(l==r){
tre[a]+=tre[b];
return a;
}
int mid=(l+r)>>1;
sl[a]=merge(sl[a], sl[b], l, mid);
sr[a]=merge(sr[a], sr[b], mid+1, r);
return a;
}
int n,m,w[MAXN],rot1[MAXN],rot2[MAXN];
int ans[MAXN];
void solve(int u, int fa){
for(int i=head[u];i;i=nxt[i]){
int v=vv[i];
if(v==fa) continue;
solve(v, u);
rot1[u]=merge(rot1[u], rot1[v], 1, n*2);
rot2[u]=merge(rot2[u], rot2[v], 1, n*2);
}
ans[u]=query(rot1[u], 1, n*2, dep[u]+w[u])+query(rot2[u], 1, n*2, n+w[u]-dep[u]);
}
int main(){
n=read(),m=read();
for(int i=1;i<n;++i){
int u=read(),v=read();
add_egde(u, v);
}
dfs(1, 0);
dfs2(1, 1);
for(int i=1;i<=n;++i) w[i]=read();
for(int i=1;i<=m;++i){
int s=read(),t=read();
int tmp=lca(s, t);
change(rot1[s], 1, n*2, dep[s], 1);
change(rot1[tmp], 1, n*2, dep[s], -1);
change(rot2[t], 1, n*2, n+dep[s]-dep[tmp]*2, 1);
change(rot2[f[tmp]], 1, n*2, n+dep[s]-dep[tmp]*2, -1);
}
solve(1, 1);
for(int i=1;i<=n;++i) printf("%d ", ans[i]);
return 0;
}
by Santiego @ 2019-10-31 11:25:32
WTF 结果是树剖求lca
写炸了?!!
int lca(int a,int b){
while(topf[a]!=topf[b]){
if(dep[a]<dep[b]) swap(a,b); // 错误!!!
a=f[topf[a]];
}
if(dep[a]<dep[b]) return a;
else return b;
}
再现当年的憨批错误,判深度应该判topf[]
的深度啊!!
果然太弱了
自闭
by Santiego @ 2019-10-31 11:30:18
@雪颜 谢谢,已经找出问题了,是树剖的锅