Sorry, your browser cannot access this site
This page requires browser support (enable) JavaScript
Learn more >

Lucas定理

Lucas定理详述

内容

对于一个组合数CmnC_m^n,取模一个素数pp的值,假设n=a×p+bn=a\times p+b,m=c×p+dm=c\times p+d有:

CnmCac×Cbd(modp)C_n^m≡C_a^c\times C_b^d(\mod p)

证明

先证明一个一会儿要用到的定理:

(1+x)p1+xp(modp)(1+x)^p≡1+x^p(\mod p)

这里可以用费马小定理加以证明:

由于(1+x)p(1+x)(modp)(1+x)^p≡(1+x)(\mod p)

xpx(modp)x^p≡x(\mod p)1+xp1+x(modp)1+x^p≡1+x(\mod p)

于是得证:(1+x)p1+xp(modp)(1+x)^p≡1+x^p(\mod p)

然后设n=a×p+bn=a\times p+b,m=c×p+dm=c\times p+d

于是:(1+x)n(1+x)a×p+b(modp)(1+x)^n≡(1+x)^{a\times p+b}(\mod p)

即:

(1+x)n(1+x)a×p×(1+x)b(modp)(1+x)^n≡(1+x)^{a\times p}\times (1+x)^b(\mod p)

由之前证明的定理得:

(1+x)a×p(1+xp)a(modp)(1+x)^{a\times p}≡(1+x^p)^a(\mod p)

即:

(1+x)a×p+b(1+xp)a×(1+x)b(modp)(1+x)^{a\times p+b}≡(1+x^p)^a\times (1+x)^ b(\mod p)

由二项式定理得:

(1+x)a×p+bi=0ia(Cai×xp×i)×j=0jb(Cbj×xj)(modp)(1+x)^{a\times p+b}≡\displaystyle \sum^{i\to a}_{i=0}(C_a^i\times x^{p\times i})\times \displaystyle \sum^{j\to b}_{j=0}(C_b^j\times x^j)(\mod p)

考虑等号两边xmx^m的系数:

由二项式定理得左边xmx^m的系数为:CnmC_n^m

将右边的式子展开,显然当i,j满足p×i+j==mp\times i+j==m时为xmx^m的系数,也就是i==c,j==di==c,j==d时,也就是:

Cac×CbdC_a^c\times C_b^d

则能得到等式CnmCab×Cbd(modp)C_n^m≡C_a^b\times C_b^d(\mod p)

于是定理得到了证明。

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
#include <iostream>
#include <cmath>
#include <cstring>
#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long ll;


//计算n!%mod
ll fact(ll n,ll mod)
{
ll res=1;
for (int i = 1; i <= n; ++i) {
res=(res*i)%mod;
}
return res;
}


void exgcd(ll a,ll b,ll &x,ll &y,ll &d)
{
if(!b)
{
d=a;
x=1;
y=0;
} else{
exgcd(b,a%b,y,x,d);
y-=x*(a/b);
}
}


//用exgcd计算t模mod的逆元
ll inv(ll t,ll mod)
{
ll d,x,y;
exgcd(t,mod,x,y,d);
return d==1?(x%mod+mod)%mod:-1;
}


//计算C(n,m)%mod:C(n,m)%mod=(n!/[m!*(n-m)!])%mod=n!%mod*inv(m!,mod)%mod*inv((n-m)!,mod)%mod
ll comb(ll n,ll m,ll mod)
{
if(m<0||m>n)return 0;
return fact(n,mod)*inv(fact(m,mod),mod)%mod*inv(fact(n-m,mod),mod)%mod;
}


//lucas定理:C(n,m)%mod=C(n/mod,m/mod)*C(n%mod,m%mod)%mod
ll lucas(ll n,ll m,ll mod)
{
return m?lucas(n/mod,m/mod,mod)*comb(n%mod,m%mod,mod)%mod:1;
}

拓展Lucas定理

考虑如果p不是素数的情况,我们考虑利用唯一分解定理将p分解为若干素数的乘积,于是可以将式子分解为一组方程组,即:

Cnmr1(modp1)C_n^m≡r_1(\mod p_1)

Cnmr2(modp2)C_n^m≡r_2(\mod p_2)

Cnmr3(modp3)C_n^m≡r_3(\mod p_3)

.

.

.

Cnmrk(modpk)C_n^m≡r_k(\mod p_k)

于是这个方程组又让我们想到了中国剩余定理(CRT),于是我们就可以使用Lucas定理把r1rkr_1…r_k求出来,再用中国剩余定理求解CnmC_n^m

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
#include <iostream>
#include <cmath>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <vector>
using namespace std;
typedef long long ll;
#define MAXN 15
ll t;
ll n,m;
ll k;
ll a[MAXN];
ll r[MAXN];

//唯一分解定理将mod分解为素数
ll fj(ll n)
{
int c=0;
for(int i=2;i<=n;i++)
{
while(n%i==0)
{
a[c++]=i;
n/=i;
}
}
return c;
}

//快速乘取模
ll mul(ll a,ll b,ll mod)
{
ll res=0;
while(b)
{
if(b&1)res=(res+a)%mod;
a=(a+a)%mod;
b>>=1;
}
return res;
}
//计算n!%mod
ll fact(ll n,ll mod)
{
ll res=1;
for (int i = 1; i <= n; ++i) {
res=(res*i)%mod;
}
return res;
}
void exgcd(ll a,ll b,ll &x,ll &y,ll &d)
{
if(!b)
{
d=a;
x=1;
y=0;
} else{
exgcd(b,a%b,y,x,d);
y-=x*(a/b);
}
}
//用exgcd计算t模mod的逆元
ll inv(ll t,ll mod)
{
ll d,x,y;
exgcd(t,mod,x,y,d);
return d==1?(x%mod+mod)%mod:-1;
}
//计算C(n,m)%mod:C(n,m)%mod=(n!/[m!*(n-m)!])%mod=n!%mod*inv(m!,mod)%mod*inv((n-m)!,mod)%mod
ll comb(ll n,ll m,ll mod)
{
if(m<0||m>n)return 0;
return fact(n,mod)*inv(fact(m,mod),mod)%mod*inv(fact(n-m,mod),mod)%mod;
}
//lucas定理:C(n,m)%mod=C(n/mod,m/mod)*C(n%mod,m%mod)%mod
ll lucas(ll n,ll m,ll mod)
{
return m?lucas(n/mod,m/mod,mod)*comb(n%mod,m%mod,mod)%mod:1;
}

//中国剩余定理求解C(n,m)%mod=C(n,m)%(a1*a2*a3...)(ai为素数)
ll china(ll n,ll *d,ll *m)
{
ll M = 1, ret = 0;
for(int i = 0; i < n; i ++) M *= m[i];
for(int i = 0; i < n; i ++)
{
ll w = M / m[i];
ret = (ret + mul(w * inv(w, m[i]), d[i], M)) % M;
}
return (ret + M) % M;
}
int main()
{
cin>>t;
while(t--)
{
cin>>n>>m>>k;
ll num=fj(k);//分解模k
for (int i = 0; i <num; ++i) {
r[i]=lucas(n,m,a[i]);
}
cout<<china(num,r,a)<<endl;
}
return 0;
}

评论