天天爱跑步 线段树合并跪调75分求助

P1600 [NOIP2016 提高组] 天天爱跑步

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

@雪颜 谢谢,已经找出问题了,是树剖的锅


|