2015 ICL, Finals, Div. 1 Ceizenpok’s formula && Gym - 100633J

上来先感谢大佬的文章:https://blog.csdn.net/clove_unique/article/details/54571216

题目链接:http://codeforces.com/gym/100633/problem/J

题意:

Ckn%p C n k % p
的值,其中 1n1018 1 ≤ n ≤ 10 18 , 0kn 0 ≤ k ≤ n , 2p1000000 2 ≤ p ≤ 1000000 .

分析:

若p为质数,可以用lucas定理直接得,但题目中p不一定为质数,则需要用拓展lucas定理+中国剩余定理。

结论:

拓展lucas定理:

可求方程:

ansc(modpk11) a n s ≡ c ( m o d p 1 k 1 )

中国剩余定理(又名孙子定理)

可求

ansc1(modm1)ansc2(modm2)....anscq(modmq) { a n s ≡ c 1 ( m o d m 1 ) a n s ≡ c 2 ( m o d m 2 ) . . . . a n s ≡ c q ( m o d m q )

的最小正整数解,解为
(i=1qciMmiinv(Mmi,mi))%M ( ∑ i = 1 q ∗ c i ∗ M m i ∗ i n v ( M m i , m i ) ) % M
其中
M=1qmi M = ∏ 1 q m i

题解:

对于p,将它表示成

p=pk11pk22...pkqq p = p 1 k 1 ∗ p 2 k 2 . . . p q k q
的形式,对于每个 pi p i ,用lucas定理求解,再用中国剩余定理求和。

唯一的问题在于在求

n!(nm)!m!%pkii n ! ( n − m ) ! m ! % p i k i
的时候,由于分母与 pkii p i k i 不一定互质,没法求逆元,所以需要先把分子分母中 pi p i 全部除去,最后再乘回来。
代码实现:

long long x=0,ans;
for(long long i=n;i;i/=pi) x+=i/pi;
for(long long i=k;i;i/=pi) x-=i/pi;
for(long long i=n-k;i;i/=pi) x-=i/pi;
ans=a*mod_inverser(b,pk)%pk*mod_inverser(c,pk)*pow_mod(pi,x,pk)%pk;

解释一下 ipi i p i 的含义:
以求 n! n ! 为例:假设 n=100,pi=2 n = 100 , p i = 2
100!=1009998...1 100 ! = 100 ∗ 99 ∗ 98 ∗ . . . ∗ 1
=(999795...1)250(123...50) = ( 99 ∗ 97 ∗ 95 ∗ . . . ∗ 1 ) ∗ 2 50 ∗ ( 1 ∗ 2 ∗ 3 ∗ . . . ∗ 50 )
=(999795...1)250(13...49)225(12...25) = ( 99 ∗ 97 ∗ 95 ∗ . . . ∗ 1 ) ∗ 2 50 ∗ ( 1 ∗ 3 ∗ . . . ∗ 49 ) ∗ 2 25 ∗ ( 1 ∗ 2 ∗ . . . ∗ 25 )
……
所以,一直除 pi p i 可将 n! n ! 表示为

pxia p i x ∗ a
a为其余的项的积。通过这个方法可求得 n!(nm)!m!%pkii n ! ( n − m ) ! m ! % p i k i

代码:

#include <iostream>
#include <bits/stdc++.h>
using namespace std;

long long exgcd(long long a,long long b,long long &x,long long &y)
{
    long long d=a;
    if(b!=0)
    {
        d=exgcd(b,a%b,y,x);
        y-=(a/b)*x;
    }else{
        x=1;
        y=0;
    }
    return d;
}
long long pow_mod(long long a,long long b,long long p)
{
    long long res=1;
    while(b)
    {
        if(b&1) res=res*a%p;
        a=a*a%p;
        b>>=1;
    }
    return res;
}
long long mod_inverser(long long a,long long m)
{
    long long x,y;
    exgcd(a,m,x,y);
    return (m+x%m)%m;
}
long long cal(long long n,long long pi,long long pk)
{
    if(n==0) return 1;
    long long ans=1;
    if(n/pk)
    {
         for(long long i=2;i<=pk;i++)
         {
             if(i%pi)
                ans=ans*i%pk;
         }
         ans=pow_mod(ans,n/pk,pk);
    }
    for(long long i=2;i<=n%pk;i++)
    {
        if(i%pi)
            ans=ans*i%pk;
    }
    return ans*cal(n/pi,pi,pk)%pk;
}
long long C(long long n,long long k,long long p,long long pi,long long pk)
{
    if(k>n) return 0;
    long long a=cal(n,pi,pk),b=cal(k,pi,pk),c=cal(n-k,pi,pk);
    long long x=0,ans;
    for(long long i=n;i;i/=pi) x+=i/pi;
    for(long long i=k;i;i/=pi) x-=i/pi;
    for(long long i=n-k;i;i/=pi) x-=i/pi;
    ans=a*mod_inverser(b,pk)%pk*mod_inverser(c,pk)*pow_mod(pi,x,pk)%pk;
    return ans*(p/pk)%p*mod_inverser(p/pk,pk)%p;
}

int main()
{
    long long n,k,p;
    cin>>n>>k>>p;
    long long ans=0;
    for(long long x=p,i=2;i<=p;i++)
    {
        if(x%i==0)
        {
            long long pk=1;
            while(x%i==0)
            {
                pk*=i;
                x/=i;
            }
            ans=(ans+C(n,k,p,i,pk))%p;
        }
    }
    cout<<ans<<endl;
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值