其他分享
首页 > 其他分享> > P4178 Tree(点分治+树状数组/容斥)

P4178 Tree(点分治+树状数组/容斥)

作者:互联网

https://www.luogu.com.cn/problem/P4178


思路:

对u遍历每一棵子树,计算出dis,并询问前面子树有多少个点深度小于等于k−d[i] ,查询有多少值<=k-d[i]的,树状数组就好了。询问结束后把这个子树的答案累加进数组。遍历完所有子树后清空当前u的所有子树答案。注意是Dfs清空。

点分治后复杂度O(nlognlogn)

#include<iostream>
#include<vector>
#include<queue>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<cstdio>
#include<algorithm>
#define debug(a) cout<<#a<<"="<<a<<endl;
#define lowbit(x) x&(-x)
using namespace std;
const int maxn=4e4+1000;
typedef int LL;
inline LL read(){LL x=0,f=1;char ch=getchar();	while (!isdigit(ch)){if (ch=='-') f=-1;ch=getchar();}while (isdigit(ch)){x=x*10+ch-48;ch=getchar();}
return x*f;}
LL n,k;
struct Edge{LL to,w;};
vector<Edge>g[maxn];
LL tree[maxn],d[maxn];
LL rt,siz[maxn],son[maxn],sum;
bool vis[maxn];
void add(LL x,LL d){ while(x<maxn){tree[x]+=d;x+=lowbit(x);}}
LL getsum(LL x){LL sum=0;while(x>0){sum+=tree[x];x-=lowbit(x);}return sum;}
void getroot(LL u,LL fa){
        siz[u]=1;son[u]=0;
        for(LL i=0;i<g[u].size();i++){
            LL v=g[u][i].to;
            if(vis[v]||v==fa) continue;
            getroot(v,u);
            siz[u]+=siz[v];
            if(siz[v]>son[u]) son[u]=siz[v];
        }
        son[u]=max(son[u],sum-siz[u]);
        if((son[u]<<1)<=sum) rt=u;
}
void dfs_res(LL u,LL fa,LL &res){
     if(d[u]<=k) res+=getsum(k-d[u])+1;///+1是本身的链
     for(LL i=0;i<g[u].size();i++){
         LL v=g[u][i].to;LL cost=g[u][i].w;
         if(vis[v]||v==fa) continue;
         d[v]=d[u]+cost;
         dfs_res(v,u,res);
     }
}
void dfs_update(LL u,LL fa,LL val){
     if(d[u]<=k&&d[u]) add(d[u],val);
     for(LL i=0;i<g[u].size();i++){
         LL v=g[u][i].to;
         if(vis[v]||v==fa) continue;
         dfs_update(v,u,val);
     }
}
LL solve(LL u,LL fa){
        d[u]=0;
        LL res=0;
        for(LL i=0;i<g[u].size();i++){
           LL v=g[u][i].to;LL w=g[u][i].w;
           if(vis[v]) continue;
           d[v]=w;
           dfs_res(v,u,res);
           dfs_update(v,u,1);///添加子树信息
        }
        dfs_update(u,fa,-1);///清空树状数组
        return res;
}
LL divide(LL u,LL fa){
        LL ans=0;
        vis[u]=true;
        ans+=solve(u,fa);///先处理以u为根的树
        for(LL i=0;i<g[u].size();i++){
            LL v=g[u][i].to;LL w=g[u][i].w;
            if(vis[v]) continue;
            rt=0;///重心重新置为0
            son[rt]=sum=siz[v];
            getroot(v,0);
            getroot(rt,0);
            ans+=divide(rt,u);///分治下去
        }
        return ans;
}
int main(void){
        n=read();
        for(LL i=1;i<=n-1;i++){
            LL u,v,w;u=read();v=read();w=read();///w++;///防止树状数组统计边权为0
            g[u].push_back({v,w});
            g[v].push_back({u,w});
        }
        k=read();///k++;
        son[0]=sum=n;
        getroot(1,0);
        getroot(rt,0);
        LL ans=0;
        ans=divide(rt,0);
        printf("%d\n",ans);

        return 0;
}


 

标签:Tree,LL,容斥,son,sum,maxn,siz,include,P4178
来源: https://blog.csdn.net/zstuyyyyccccbbbb/article/details/116502388