组合数C(n,m)的四种求解方法

转自:文章

1、暴力求解

C(n,m)=n*(n-1)*...*(n-m+1)/m!,(n<=15);

int CF(int n,int m)
{
    int ans=1,i,j;
    for(i=n;i>=n-m+1;i--)
    ans*=i;
    for(i=m;i>=2;i--) 
    ans/=i;
    return ans;
}

 

2、打表

C(n,m)=C(n-1,m-1)+C(n-1,m),(n<=10000);

const int maxn = 10010;
const int MOD = 100007;
void CF(int n,int m)
{
    int i,j;
    for(i=0;i<=maxn;i++)
    {
        c[0][i]=0;c[i][0]=1;
    }
    for(i=1;i<=maxn;i++)
        for(j=1;j<=maxn;j++)
        c[i][j]=(c[i-1][j]+c[i-1][j-1])%MOD;
}

 

3、质因数分解

C(n,m)=n!/(m!*(n-m)!),C(n,m)=p1a1-b1-c1p2a2-b2-c2…pkak-bk-ck,(n<=10000000)

#include<iostream>
#include<cstdio>
#include<vector>
using namespace std;
const int MOD = 100007;
const int maxn = 1000001;
bool a[maxn]={false};
vector <int> prim_produce() //生成素数序列 
{
    vector <int> vc;
    vc.push_back(2);
    int i,j;
    for(i=3;i*i<=maxn;i+=2)
    {
        if(!a[i])
        {
            vc.push_back(i);
            for(j=i*i;j<=maxn;j+=i)
            {
                a[j]=true;
            }
        }
    }
    while(i<maxn)
    {
        if(!a[i]) vc.push_back(i);
        i+=2;
    }
    return vc;
}
//计算n!素数p的指数 
int cal(int x,int p)
{
    int ans=0;
    long long re=p;
    while(x>=re)
    {
        ans+=x/re;
        re*=p;
    }
    return ans; 
}

int Pow(long long n,int k) //二分求n的k次方 
{
    long long ans=1;
    while(k)
    {
        if(k&1) ans=ans*n%MOD;
        n=(n*n)%MOD;
        k>>=1;
    }
    return ans;
}
int comb(int n,int m) //计算公式 
{
    vector <int> prim=prim_produce();
    long long ans=1;
    int num;
    for(int i=0;i<prim.size()&&prim[i]<=n;i++)
    {
        num=cal(n,prim[i])-cal(m,prim[i])-cal(n-m,prim[i]);
        ans=(ans*Pow(prim[i],num))%MOD;
    }
    return ans;
}
int main(void)
{
    int n,m;
    while(~scanf("%d%d",&n,&m))
    {
        printf("%d\n",comb(n,m));
    }
    return 0;
}
View Code

 

4、Lucas定理

将m,n化为p进制,有:C(n,m)=C(n0,m0)*C(n1,m1)...(mod p),算一个不是很大的C(n,m)%p,p为素数,化为线性同余方程,用扩展的欧几里德定理求解,n在int范围内,修改一下可以满足long long范围内。

 

 

#include <stdio.h>
const int M = 2013;
int ff[M+5];  //打表,记录n!,避免重复计算
 
//求最大公因数
int gcd(int a,int b)
{
    if(b==0)
        return a;
    else
        return gcd(b,a%b);
}
 
//解线性同余方程,扩展欧几里德定理
int x,y;
void Extended_gcd(int a,int b)
{
    if(b==0)
    {
        x=1;
        y=0;
    }
    else
    {
        Extended_gcd(b,a%b);
        long t=x;
        x=y;
        y=t-(a/b)*y;
    }
}
 
//计算不大的C(n,m)
int C(int a,int b)
{
    if(b>a)
        return 0;
    b=(ff[a-b]*ff[b])%M;
    a=ff[a];
    int c=gcd(a,b);
    a/=c;
    b/=c;
    Extended_gcd(b,M);
    x=(x+M)%M;
    x=(x*a)%M;
    return x;
}
 
//Lucas定理
int Combination(int n, int m)
{
    int ans=1;
    int a,b;
    while(m||n)
    {
        a=n%M;
        b=m%M;
        n/=M;
        m/=M;
        ans=(ans*C(a,b))%M;
    }
    return ans;
}
 
int main()
{
    int i,m,n;
    ff[0]=1;
    for(i=1; i<=M; i++) //预计算n!
        ff[i]=(ff[i-1]*i)%M;
    while(~scanf("%d%d",&n, &m))
    {
        printf("%d\n",Combination(n,m));
    }
    return 0;
}
View Code

 

 

 

posted @ 2018-12-26 14:19  麟阁  阅读(2568)  评论(0编辑  收藏  举报