码迷,mamicode.com
首页 > 其他好文 > 详细

Lucas定理--组合数取模

时间:2019-10-27 12:19:23      阅读:60      评论:0      收藏:0      [点我收藏+]

标签:ace   namespace   cas   int   cin   mod   space   str   efi   

Lucas定理是用来求 c(n,m) mod p,p为素数的值。

C(n,m)%p = C(n/p,m/p) * C(n%p,m%p)%p 

\(Lucas(n,m,p)=C(n \% p,m \% p) \times Lucas(\frac{n}{p},\frac{m}{p},p)\)

\(\binom{n}{m}=\frac{n!}{m!(n-m)!}\)

卢卡斯定理模板

#include <cstdio>
#include <iostream>
#define ll long long
using namespace std;
ll pow(ll a,ll b,ll p){
    ll ans = 1;
    a%=p;
    while(b){
        if(b&1)ans = (ans%p)*(a%p)%p;
        b>>=1;
        a = a*a%p;
    }
    return ans;
}
ll inv(ll x,ll p){//x关于p的逆元,p为素数
    return pow(x,p-2,p);
}
ll C(ll n,ll m,ll p){//求组合数C(n,m)%p
    if(n<m)return 0;
    ll up=1,down=1;//分子分母
    for(ll i=n-m+1;i<=n;i++)up=up*i%p;
    for(ll i=1;i<=m;i++)down=down*i%p;
    return up*inv(down,p)%p;
}
ll lucas(ll n,ll m,ll p){
    if(m==0)return 1;
    return C(n%p,m%p,p)*lucas(n/p,m/p,p)%p;
}
int main(){
    int t;
    cin>>t;
    ll n,m,p;
    while(t--){
        cin>>n>>m>>p;
        cout<<lucas(n,m,p)<<endl;
    }
    return 0;
}

扩展卢卡斯定理

当p不为素数的时候

Lucas定理--组合数取模

标签:ace   namespace   cas   int   cin   mod   space   str   efi   

原文地址:https://www.cnblogs.com/Emcikem/p/11746942.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!