zoukankan      html  css  js  c++  java
  • [bzoj2839]集合计数 题解 (组合数+容斥)

    Description

    一个有N个元素的集合有2^N个不同子集(包含空集),现在要在这2^N个集合中取出若干集合(至少一个),使得
    它们的交集的元素个数为K,求取法的方案数,答案模1000000007。(是质数喔~)

    Input

    一行两个整数N,K

    Output

    一行为答案。

    Sample Input

    3 2

    Sample Output

    6

    HINT

    【样例说明】

    假设原集合为{A,B,C}

    则满足条件的方案为:{AB,ABC},{AC,ABC},{BC,ABC},{AB},{AC},{BC}

    【数据说明】

         对于100%的数据,1≤N≤1000000;0≤K≤N;

    首先在1~n中取k个数来当交集,方案数显然为$C_n^k$

    记得$n-=k$

     之后应该还要乘上一坨奇怪的东西

    我们把包含一个特定元素的所有方案丢到一个集合中

    那么会有n个集合卡在一起

    那么我们求得就是这张鬼xu的图中的无交集部分

    该部分$=ALL-part_{>=1}+part_{>=2}-part_{>=3}+...$

    而至少有i个元素作为交集的方案数为$C_{n}^{i}(2^{2^{n-i}}-1)$

    这个式子怎么求出来的呢?

    首先任取i个数作为交集

    剩下$n-i$个数 (不能都不选)能组成$2^{n-i}$个集合

    然后从这些集合中选组成新集合

    $ans=sum_{i=0}^n (-1)^i imes C_{n}^i imes (2 ^ {2 ^ {(n - i)}} - 1)$

    2的次幂那部分可以递推求 (快速幂这么粗鲁的方式我才不会用!)

    #include<cstdio>
    #include<iostream>
    #include<cstring>
    using namespace std;
    const int mod=1e9+7,N=1000005;
    int n,k,fac[N],inv[N];
    inline int qpow(int a,int b)
    {
        int res=1;
        for( ;b;b>>=1,a=1LL*a*a%mod)
            if(b&1)res=1LL*res*a%mod;
        return res;
    }
    inline int C(int x,int y)
    {
        if(x<0||y<0||x<y)return 0;
        return (1LL*inv[y]*inv[x-y]%mod)*fac[x]%mod;
    }
    inline int num(int a)
    {
        return (a&1)?-1:1;
    }
    int main()
    {
        scanf("%d%d",&n,&k);
        fac[0]=1;
        for(int i=1;i<=n;i++)
            fac[i]=1LL*fac[i-1]*i%mod;
        inv[n]=qpow(fac[n],mod-2); 
        for(int i=n-1;i>=0;i--)
            inv[i]=1LL*inv[i+1]*(i+1)%mod;
        int com=C(n,k),ans=0;
        int tmp,mii=2;n-=k;
        for(int i=n;i>=0;i--)
        {
            tmp=1LL*C(n,i)*num(i)*(mii-1)%mod;
            mii=1LL*mii*mii%mod;
            ans+=tmp,ans%=mod;
        }
        ans=1LL*ans*com%mod;
        printf("%d
    ",(ans+mod)%mod);
        return 0;
    }

     

  • 相关阅读:
    pcie dma
    Virtual Network
    hashcode与equals
    java四类八种基本数据类型
    jdk 1.7 LinkedList 源码分析
    jdk 1.7 ArrayList 源码分析
    httpclient发送xml字符串(推送)
    HttpClient使用Post和Get提交参数
    httpClient发送Json请求,结果返回Json.
    HttpClient
  • 原文地址:https://www.cnblogs.com/Rorschach-XR/p/11131690.html
Copyright © 2011-2022 走看看