zoukankan      html  css  js  c++  java
  • 51NOD 1709:复杂度分析——题解

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1709

    (我什么时候看到二进制贡献才能条件反射想到按位处理贡献呢……)

    参考:https://www.cnblogs.com/hzoier/p/8593825.html

    最朴素当然是LCA暴算。

    但是更聪明的做法是考虑每个节点$u$,固定$lca$求贡献,能够$LCA(u,v)=lca$的点$v$的个数显然为$size[lca]-size[u$所在的以$lca$为根的子树$]$。

    于是答案也就出来了,就是$bit[u->lca]*(size[lca]-size[u$所在的以$lca$为根的子树$])$。

    当然复杂度不理想,因为有很多节点的贡献我们应该是可以一起算的。于是我们按照二进制位每位计算贡献。

    如果你画个图你就会发现,设$anc[i][j]$为$i$的第$2^j-1$个祖先(注意,不是第$2^j$个祖先,原因请画图理解),则对于$u$,其第$j$位所带来的贡献就是$size[anc[u][j+1]]-size[anc[u][j]]$。

    当然还没完,在这之上还有能够为其做出第j位贡献的点,所以我们还要加上$w[fa[anc[u][j+1]]]$才行。

    (强烈建议画图体验,仅凭文字很难说明。)

    (其实也就是$2^{j+1}$及以上有贡献的点,我们可以通过u一次跳跃过去,然后就是递归了)

    #include<map>
    #include<cmath>
    #include<stack>
    #include<queue>
    #include<cstdio>
    #include<cctype>
    #include<vector>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    typedef long long ll;
    const int N=1e5+5;
    const int INF=1e9;
    const int B=17;
    inline int read(){
        int X=0,w=0;char ch=0;
        while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
        while(isdigit(ch))X=(X<<3)+(X<<1)+(ch^48),ch=getchar();
        return w?-X:X;
    }
    struct node{
        int to,nxt;
    }e[N*2];
    int n,cnt,head[N],anc[N][B+4],fa[N];
    int size[N],sum[N],idx[N],tot;
    ll w[N];
    inline void add(int u,int v){
        e[++cnt].to=v;e[cnt].nxt=head[u];head[u]=cnt;
    }
    void dfs(int u){
        idx[++tot]=u;size[u]=1;
        for(int i=head[u];i;i=e[i].nxt){
            int v=e[i].to;
            if(v==fa[u])continue;
            fa[v]=u;dfs(v);size[u]+=size[v];
        }
    }
    int main(){
        n=read();
        for(int i=1;i<n;i++){
            int u=read(),v=read();
            add(u,v);add(v,u);
        }
        dfs(1);
        for(int i=1;i<=n;i++)
            anc[i][0]=i,anc[i][1]=fa[i];
        for(int j=2;j<=B+1;j++){
            for(int i=1;i<=n;i++){
                anc[i][j]=fa[anc[anc[i][j-1]][j-1]];
            }
        }
        ll ans=0;
        for(int j=0;j<=B;j++){
            memset(w,0,sizeof(w));
            for(int i=1;i<=n;i++){
                int u=idx[i],v1=anc[u][j+1],v2=anc[u][j];
                if(!v1)v1=1;if(!v2)v2=1;
                w[u]=size[v1]-size[v2];
                w[u]+=w[fa[anc[u][j+1]]];
                ans+=w[u];
            }
        }
        printf("%lld
    ",ans);
        return 0;
    }

    +++++++++++++++++++++++++++++++++++++++++++

    +本文作者:luyouqi233。               +

    +欢迎访问我的博客:http://www.cnblogs.com/luyouqi233/ +

    +++++++++++++++++++++++++++++++++++++++++++

  • 相关阅读:
    数据仓库系列之ETL过程和ETL工具
    大数据从入门到精通
    Mysql误删表中数据与误删表的恢复方法(转)
    Vscode 小白使用介绍
    Window 通过cmd查看端口占用,杀死进程方法
    Python基础教程,Python入门教程(非常详细)
    tuxedo安装与配置入门
    Win10系统Ping端口及利用telnet命令Ping 端口
    Tuxedo 介绍
    hibernate框架的简单入门
  • 原文地址:https://www.cnblogs.com/luyouqi233/p/9210204.html
Copyright © 2011-2022 走看看