【学习笔记】扩展卢卡斯定理 exLucas
引子
求
\]
不保证 \(p\) 是质数。
正文
对于传统的 Lucas 定理,必须要求 \(p\) 是质数才行。若 \(p\) 不一定是质数,则需要扩展 Lucas 定理
前置知识
扩展欧几里得和中国剩余定理。
算法内容
将 \(p\) 用唯一分解定理分解,即
\]
若求出了
\]
就可以用中国剩余定理合并答案了。那么此时我们要求的就是
\]
但是显然 \(m!\) 和 \((n-m)!\) 在 \(\text{mod}\ p^k\) 的意义下不一定存在逆元。
对于两个数 \(a\) 和 \(p\),存在逆元的充要条件是 \(\gcd(a,p)=1\)。
所以我们可以强行让他们互质。令 \(x,y,z\) 分别为 \(n!,m!,(n-m)!\) 中 \(p\) 因子的个数,我们可以将原式转化得到
\]
这样我们就可以求逆元了,那么问题就转化成了对于每一个数 \(n\) ,求出 \(\cfrac{n!}{p^x}\ \text{mod}\ p^k\) 即可。
将 \(n!\) 展开
\]
其中左边的括号内是 \(p\) 的所有倍数。
易知,在 \(1\) ~ \(n\) 中,有 \(\bigg\lfloor\cfrac{n}{p}\bigg\rfloor\) 个 \(p\) 的倍数。因此继续变形原式
\]
\]
然后你就发现他可以套娃了,把第三项拆成能整除之前和余数两个部分
\]
也就是说上面这个式子是有循环节的,其中第三项是循环节 \(1\) ~ \(p\) 中非 \(p\) 倍数的乘积,而第四项则是 \(p\) 以后的。
前面这个 \(p^{\lfloor\frac{n}{p}\rfloor}\) 是要除掉的,但是别忘了 $ \bigg(\bigg\lfloor\cfrac{n}{p}\bigg\rfloor\bigg)!$ 中可能也有 \(p\) 因子。因此我们定义 \(f(n)=\cfrac{n}{p^x}\),得到
\]
且
\]
这样就可以 \(O(n\log p)\) 递推了。
回到最开始的式子
\]
此时 \(f(m)\) 一定和 \(p^k\) 互质,我们可以用扩展欧几里得求逆元来得到答案了。
现在我完全明白了!
但是还剩一个 \(p^{x-y-z}\) 没算。比如,我们要算出现在 \(f(n)\) 的 \(x\)。设 \(g(n)=x\),再看这个式子
\]
我们要求的部分就是 \(p^{\lfloor\frac{n}{p}\rfloor}\) 且乘上 \(\bigg(\bigg\lfloor\cfrac{n}{p}\bigg\rfloor\bigg)!\) 中的 \(p\) 因子。
因此我们得到递推式
\]
且
\]
所以最后的结论就是
\]
然后用中国剩余定理合并答案就可以得到结果了!
代码
#include <bits/stdc++.h>
#define int long long
typedef long long ll;
const int maxn=1e3+10;
using namespace std;
inline int read(){
int x=0;bool fopt=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar())if(ch=='-')fopt=0;
for(;isdigit(ch);ch=getchar())x=(x<<3)+(x<<1)+ch-48;
return fopt?x:-x;
}
inline int qpow(int x,int b,int p){
int ans=1,base=x;
while(b){
if(b&1)ans=ans*base%p;
base=base*base%p;
b>>=1;
}
return ans;
}
void exgcd(int a,int b,int &x,int &y){
if(!b)return x=1,y=0,void();
exgcd(b,a%b,x,y);
int z=x;x=y;y=z-a/b*y;
}
inline int inv(int a,int p){
int x=0,y=0;
exgcd(a,p,x,y);
return (x+p)%p;
}
int F(int n,int p,int P){
if(!n)return 1;
int rou=1,rem=1;//rou表示循环节中的,rem表示余数部分的
for(int i=1;i<=P;i++)
if(i%p)rou=rou*i%P;
rou=qpow(rou,n/P,P);
for(int i=P*(n/P);i<=n;i++)
if(i%p)rem=rem*(i%P)%P;
return F(n/p,p,P)*rou%P*rem%P;
}
int G(int n,int p){
if(n<p)return 0;
return (n/p)+G(n/p,p);
}
inline int C(int n,int m,int p,int P){
int x=F(n,p,P),y=inv(F(m,p,P),P),z=inv(F(n-m,p,P),P);
int Pow=qpow(p,G(n,p)-G(m,p)-G(n-m,p),P);
return x*y%P*z%P*Pow%P;
}
int tot;
int A[maxn],B[maxn];
inline void Init(int n,int m,int x){
int M=sqrt(x);
for(int i=2;i<=M;i++){
if(x%i==0){
int P=1;
while(x%i==0){
P*=i;x/=i;
}
A[++tot]=P;B[tot]=C(n,m,i,P);//算出每一个p^k意义下的答案
}
}
if(x!=1){
A[++tot]=x;B[tot]=C(n,m,x,x);
}
}
inline int exLucas(int n,int m,int p){
Init(n,m,p);
int ans=0;
for(int i=1;i<=tot;i++){
int u=p/A[i],v=inv(u,A[i]);//普通的CRT合并答案
ans=(ans+B[i]*u%p*v%p)%p;
}
return ans;
}
signed main(){
int n=read(),m=read(),p=read();
printf("%lld\n",exLucas(n,m,p));
return 0;
}
例题:礼物
一年一度的圣诞节快要来到了。每年的圣诞节小 E 都会收到许多礼物,当然他也会送出许多礼物。不同的人物在小 E 心目中的重要性不同,在小 E 心中分量越重的人,收到的礼物会越多。
小 E 从商店中购买了 \(n\) 件礼物,打算送给 \(m\) 个人,其中送给第 \(i\) 个人礼物数量为 \(w_i\)。请你帮忙计算出送礼物的方案数(两个方案被认为是不同的,当且仅当存在某个人在这两种方案中收到的礼物不同)。由于方案数可能会很大,你只需要输出模 \(P\) 后的结果。\(P\) 不保证是质数。若无法满足条件,输出
Impossible。
怎么这么多屑题叫礼物啊(恼)
显然答案是 \(\prod\limits_{i=1}^mC_{n-\sum_{j=1}^{i-1}w_j}^{w_i}\pmod P\),然后用扩展卢卡斯求解就可以了。至于无解情况,只要看剩余的礼物数是否为负即可。
【学习笔记】扩展卢卡斯定理 exLucas的更多相关文章
- [学习笔记]扩展LUCAS定理
可以先做这个题[SDOI2010]古代猪文 此算法和LUCAS定理没有半毛钱关系. [模板]扩展卢卡斯 不保证P是质数. $C_n^m=\frac{n!}{m!(n-m)!}$ 麻烦的是分母. 如果互 ...
- 扩展卢卡斯定理(Exlucas)
题目链接 戳我 前置知识 中国剩余定理(crt)或扩展中国剩余定理(excrt) 乘法逆元 组合数的基本运用 扩展欧几里得(exgcd) 说实话Lucas真的和这个没有什么太大的关系,但是Lucas还 ...
- 【知识总结】扩展卢卡斯定理(exLucas)
扩展卢卡斯定理用于求如下式子(其中\(p\)不一定是质数): \[C_n^m\ mod\ p\] 我们将这个问题由总体到局部地分为三个层次解决. 层次一:原问题 首先对\(p\)进行质因数分解: \[ ...
- bzoj2142 礼物——扩展卢卡斯定理
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2142 前几天学了扩展卢卡斯定理,今天来磕模板! 这道题式子挺好推的(连我都自己推出来了) , ...
- [笔记] 扩展Lucas定理
[笔记] 扩展\(Lucas\)定理 \(Lucas\)定理:\(\binom{n}{m} \equiv \binom{n/P}{m/P} \binom{n \% P}{m \% P}\pmod{P} ...
- 卢卡斯定理&扩展卢卡斯定理
卢卡斯定理 求\(C_m^n~mod~p\) 设\(m={a_0}^{p_0}+{a_1}^{p_1}+\cdots+{a_k}^{p_k},n={b_0}^{p_0}+{b_1}^{p_1}+\cd ...
- LG4720 【模板】扩展卢卡斯定理
扩展卢卡斯定理 求 \(C_n^m \bmod{p}\),其中 \(C\) 为组合数. \(1≤m≤n≤10^{18},2≤p≤1000000\) ,不保证 \(p\) 是质数. Fading的题解 ...
- 【学习笔记】Polya定理
笔者经多番周折终于看懂了\(\text{Burnside}\)定理和\(\text{Polya}\)定理,特来写一篇学习笔记来记录一下. 群定义 定义:群\((G,·)\)是一个集合与一个运算·所定义 ...
- [Android学习笔记]扩展application
扩展Application对象 每一个应用程序启动之后,都会分配一个linux用户,并且运行在一个独立的进程中.默认情况下,一个应用程序只会运行在一个进程中(可以通过配置android:process ...
随机推荐
- H5游戏定制,4大优势助力企业曝光10W+
H5游戏定制,4大优势助力企业曝光10W+ 移动互联网已成为了人们生活的一部分,普通广告形式已很难吸引用户的眼球,企业要怎样才能将广告更广泛的传播给更多用户呢?根据TOM游戏多年从业经验,为大家分享以 ...
- oracle之三rman 不完全恢复
rman 不完全恢复 9.1 rman 不完全恢复的三个标准模式:基于time.基于scn和基于sequence: 范例1:恢复过去某个时间点误操作,一般使用基于time或scn. 1)环境:有一套全 ...
- v-if和v-show的区别与使用
1.共同点: v-if 和 v-show 都能实现元素的显示隐藏. 2.不同点: v-if显示隐藏是将dom元素整个添加或删除,v-show元素隐藏时,会在dom节点中把该元素设置css属性为disp ...
- powershell编程实例-001-生成指定大小的null/zero文件或随机文件
NULL文件,也有的称为zero文件,即全是二进制/十六进制的0文件 在powershell 中可以按如下方法生成指定大小的zero文件: 只需要修改大小即可,格式如3MB,或者2GB $tempFi ...
- PHP绕过disable_function
PHP绕过disable_function 常规绕过 exec exec执行command命令,但是不会输出全部结果,而是返回结果的最后一行. 想得到全部的结果,可以使用第二个参数,让其输出到一个数组 ...
- ISCC2018 writeup(web)
比较数字大小 F12 修改maxlength为4 web01 strcmp()函数遇到数组会返回NULL 而PHP是弱类型语言 在==比较的时候,如果有数值的话会先将字符串转换为数值在进行比较,而N ...
- IIS目录浏览模式打开文件还是无法下载
写在前面的话 IIS已经设置目录浏览启用,且可以正常访问到文件,说明这些设置没问题,但是点击文件进行下载时,却提示无法下载,文件不存在等等,有的又可以,一顿操作后发现,原来是文件类型没有包含在MIME ...
- Spring AOP系列(二) — 动态代理引言
接上一篇Spring AOP系列(一)- 代理模式,本篇来聊聊动态代理. 动态代理与静态代理的区别 要想了解动态代理与静态代理的区别,需要有两个前置知识点:java程序是如何执行的以及类加载机制. j ...
- Java单播、组播(多播)、广播的简单实现
简介 单播有TCP和UDP两种实现,组播(多播)和广播只有UDP一种实现.单播和广播基本一样,只是广播的数据包IP为广播IP. 单播 DatagramSocket和DatagramPacket 服 ...
- Python-对字典进行排序
案例: 某班英语成绩以字典的形式存储为: {'lili':78, 'jin':50, 'liming': 30, ......} 依据成绩高低,进行学生成绩排名 如何对字典排序? 方法1: #!/us ...