bzoj千题计划263:bzoj4870: [六省联考2017]组合数问题

http://www.lydsy.com/JudgeOnline/problem.php?id=4870

 

 

 80分暴力打的好爽 \(^o^)/~

 

预处理杨辉三角

 

 

令m=n*k

要求满足m&x==x ,x<=m, x%k==r 的x的个数

结论:若n&m==m,则C(n,m)为奇数,否则为偶数

枚举m的子集,判断是否%k==r

时间复杂度:O(m的位子集个数),即O(2^(m的二进制中1的个数))极限是O(n*k)

 

 

杨辉三角第i行的和=2^i,即

那么用2^(nk) 减去 前面不用的C

因为r<=50,所以这种C的个数<=50

暴力计算C即可

 

k=2 就是C(2n,r)+C(2n,r+2)+C(2n,r+4)……

是隔一个加一个

 杨辉三角 每一行的 奇数列之和=偶数列之和

所以 2^(2n)/2 - 前面不用的C,也是隔一个减一个

 

除2的计算 要乘2的逆元,但是p不是素数

 

所以 计算2^(2n) 改成计算 2^(2n-1)

 

 

预处理阶乘和阶乘的逆元,枚举计算C ,最多会计算n个C

 

不会,求指点

 

考虑组合数C(n,m)的实际意义:从n个元素里选出m个元素的方案数

那么本题就是求从n*k个元素里,选出%k=r个元素的方案数

dp[i][j] 表示从前i个元素里,选出%k=j 个元素的方案数

第i个不选:dp[i][j]+=dp[i-1][j]

第i个选:dp[i][j]+=dp[i-1][(j-1+k)%k]

矩阵乘法优化

#include<cstdio>
#include<cstring>

using namespace std;

int p,K;

int a[50][50],ans[50][50],f[50][50];

int C[50][50];

void mul(int A[50][50],int B[50][50])
{
    memset(C,0,sizeof(C));
    for(int i=0;i<K;++i)
        for(int j=0;j<K;++j)
            for(int k=0;k<K;++k)
                C[i][j]=(C[i][j]+1LL*A[i][k]*B[k][j]%p)%p;
    for(int i=0;i<K;++i)
        for(int j=0;j<K;++j)
            A[i][j]=C[i][j];
}

int main()
{
    int n,k,r;
    scanf("%d%d%d%d",&n,&p,&K,&r);
    for(int i=0;i<=K-2;++i) a[i][i]=a[i][i+1]=1;
    a[K-1][K-1]++; a[K-1][0]++;
    f[0][0]=1;
    for(int i=0;i<K;++i) ans[i][i]=1;
    long long m=1LL*n*K;
    for(;m;mul(a,a),m>>=1)
        if(m&1) mul(ans,a);
    mul(f,ans);
    printf("%d",f[0][r]);
}
AC代码

 

#include<cstdio>
#include<cstring>

using namespace std;

typedef long long LL;

int n,p,k,r;

int C[901][901];

int num[51];

int inv[1000001],fac[1000001];

LL xx;

void ADD(int &x,int y)
{
    xx=x;
    xx+=y;
    xx-=xx>=p ? p : 0;
    x=xx;
}

void YangHui()
{
    C[0][0]=1;
    int m=n*k;
    for(int i=1;i<=m;++i)
    {
        C[i][0]=1;
        for(int j=1;j<=m;++j) C[i][j]=(1LL*C[i-1][j]+C[i-1][j-1])%p;
    }
    int ans=0;
    for(int i=0;i<=n;++i) 
    {
        if(i*k+r>m) break;
        ADD(ans,C[m][i*k+r]);
    }
    printf("%d",ans);
}

void Yu()
{
    LL m=1LL*n*k;
    int ans=0;
    for(LL t=m;t;t=(t-1)&m)
        if(t%k==r) ans^=1;
    printf("%d",ans);
}    

int Pow(int a,int b)
{
    int res=1;
    for(;b;a=1LL*a*a%p,b>>=1)
        if(b&1) res=1LL*res*a%p;
    return res;
}

int get_gcd(int a,int b)
{
    return !b ? a : get_gcd(b,a%b);
}

int get_C(int m,int k)
{
    for(int i=m-k+1;i<=m;++i) num[i]=i;
    int x,gcd;
    for(int i=2;i<=k;++i)
    {
        x=i;
        for(int j=m-k+1;j<=m && x!=1;++j)
        {
            gcd=get_gcd(num[j],x);
            num[j]/=gcd;
            x/=gcd;
        }
    }
    int ans=1;
    for(int i=m-k+1;i<=m;++i) ans=1LL*ans*num[i];
    return ans;
}

void PreSum()
{
    int a=Pow(2,n);
    int b=0;
    for(int i=0;i<r;++i) ADD(b,get_C(n,i));
    a-=b;
    if(a<0) a+=p;
    printf("%d",a);
}

void JiOu()
{
    int a=Pow(2,n*2-1);
    int b=0;
    for(int i=r-2;i>=0;i-=2) ADD(b,get_C(n<<1,i));
    a-=b;
    if(a<0) a+=b;
    printf("%d",a);
}

int get_inv(int x)
{
    if(inv[x]!=-1) return inv[x];
    inv[x]=Pow(fac[x],p-2);
    return inv[x];
}

void Cal()
{
    int m=n*k;
    fac[0]=1;
    for(int i=1;i<=m;++i) fac[i]=1LL*fac[i-1]*i%p;
    memset(inv,-1,sizeof(inv));
    inv[0]=1;
    int h=r;
    int ans=0;
    int tmp;
    for(int i=1; h<=m;++i,h+=k)
    {
        tmp=fac[m];
        tmp=1LL*tmp*get_inv(h)%p*get_inv(m-h)%p;
        ADD(ans,tmp);
    }
    printf("%d",ans);
}

int main()
{
    freopen("problem.in","r",stdin);
    freopen("problem.out","w",stdout);
    scanf("%d%d%d%d",&n,&p,&k,&r);
    if(n<=30 && k<=30) YangHui();
     else if(p==2) Yu();
    else if(k==1) PreSum();
    else if(k==2) JiOu();
    else Cal();
    return 0;
}
80分暴力

 

脑抽错误:

C(n,m)一定是偶数,但 对p 取模之后不能保证是偶数

考试的时候没考虑这个 ,直接/2 丢了10分,~~~~(>_<)~~~~

 

 

posted @ 2018-03-07 20:31  TRTTG  阅读(445)  评论(0编辑  收藏  举报