Lucas定理

先上结论:

当p为素数:

\(\binom{ N }{M} \equiv \binom{ N/p }{M/p}*\binom{ N mod p }{M mod p} (mod p)\)

证明:令 \(s=\lfloor \frac{n}{p} \rfloor\),\(q=n\bmod p\),\(t=\lfloor \frac{m}{p} \rfloor\),\(r=m \bmod p\)。

需证明 \(\binom{sp+q}{tp+r}\equiv \binom{s}{t}\binom{q}{r} \pmod p\)。\((1+x)^n\equiv (1+x)^{sp+q}\equiv((1+x)^p)^s(1+x)^q\)。

因为 \((1+x)^p=\sum\limits_{i=0}^p \binom{p}{i} x^i\),

所以 \((1+x)^p\equiv(1+x^p)\)。

那么就有 \((1+x)^n\equiv(1+x^p)^s(1+x)^q\equiv\sum\limits_{i=0}^{s} \binom{s}{i} x^{pi}\cdot\sum\limits_{j=0}^q \binom{q}{i}x^i\)。

考虑 \(\binom{n}{m}\) 也就是 \(\binom{sp+q}{tp+r}\)

实际上是在多项式 \((1+x)^n\) 中 \(x^{tp+r}\) 项的系数,

由前面的同余式得到这个系数同时也是 \(\binom{s}{t}\binom{q}{r}\) 也就是 \(i\) 取 \(t\) \(j\) 取 \(r\) 的情况,

所以 \(\binom{n}{m}\equiv \binom{\lfloor \frac{n}{p} \rfloor}{\lfloor \frac{m}{p} \rfloor} \binom{n\bmod p}{m\bmod p} \pmod {p}\) 得证。

以上出自 ZCDHJ 的友情提供

也就是说我们对于 \(\binom{ N }{M} mod p\) 可以化为 \(\binom{ N/p }{M/p}*\binom{ N mod p }{M mod p} (mod p)\)

而 \(\binom{ N/p }{M/p}\) 显然可以继续递归调用下去

代码(可过luogu 模板):

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. #define re register
  4. #define in inline
  5. #define ll long long
  6. #define int ll
  7. #define get getchar()
  8. in int read()
  9. {
  10. int t=0; char ch=get;
  11. while(ch<'0' || ch>'9') ch=get;
  12. while(ch<='9' && ch>='0') t=t*10+ch-'0',ch=get;
  13. return t;
  14. }
  15. const int _=1e6+5;
  16. int n,m,p;
  17. ll fac[_];
  18. in ll fastpow(int a,int b) //快速幂
  19. {
  20. ll res=1;
  21. while(b)
  22. {
  23. if(b&1)
  24. res=(res*a)%p;
  25. a=a*a%p,b>>=1;
  26. }
  27. return res;
  28. }
  29. in ll C(int a,int b) //组合数
  30. {
  31. if (a < b) return 0;
  32. return fac[a]*fastpow(fac[b],p-2)%p*fastpow(fac[a-b],p-2)%p;
  33. }
  34. in ll lucas(int a,int b)//卢卡斯
  35. {
  36. if(!b) return 1;
  37. return C(a%p,b%p)*lucas(a/p,b/p)%p; //显然C(a%p,b%p)不可能再用卢卡斯化简.所以直接暴力求
  38. }//所以Lucas 通常只用于模数较小时
  39. signed main()
  40. {
  41. int T=read();
  42. while(T--)
  43. {
  44. n=read(),m=read(),p=read();
  45. n+=m;
  46. fac[0]=1;
  47. for(re int i=1;i<=p;i++)
  48. fac[i]=(fac[i-1]*i)%p;
  49. cout<<lucas(n,m)<<endl;
  50. }
  51. return 0;
  52. }

例题解析([SHOI2015]超能粒子炮·改)

传送门

题意简化

求 \(\sum\limits_{i=0}^{k} \binom{n}{i}\) mod p

p为2333

Solution

(以下用 % 表示mod,除法视作向下取整)

不妨设 \(F(n,k)=\sum\limits_{i=0}^{k} \binom{n}{i} mod p\)

根据Lucas 可推出

\(F(n,k)=\sum\limits_{i=0}^{k} \binom{n/p}{i/p} * \binom{n%p}{i%p}mod p\) (因为式子最后都要mod p,所以以下就省略了)

因为除法是向下取整的

考虑数论分块 (就是在1~n 枚举时若其中含除法,可以把每n/const 分为一块,每一块中的取值时一样的)

则有 \(F(n,k)=\sum\limits_{i=0}^{k/p} \binom{n/p}{i} * \sum\limits_{j=0}^{min(p-1,k-p*i)} \binom{n%p}{j}\)

再观察一下 min(p-1,k-pi) ,

可以发现只有 k-p
i==k%p 时才会小于 p-1,

而此时 i=k/p

所以我们又可以把式子再次拆开:

$ F(n,k)= \sum\limits_{i=0}^{k/p} \binom{n/p}{i} $ \(*\) $ \sum\limits_{j=0}^{p-1} \binom{n%p}{j}$ \(+\) $ \binom{n/p}{i/p} $ \(*\) $ \sum\limits_{j=0}^{k mod p} $ \(*\) $\binom{n%p}{j} $

$ \sum\limits_{i=0}^{k/p} \binom{n/p}{i} $ 是 F(n/p.k/p-1),

$ \sum\limits_{j=0}^{p-1} \binom{n%p}{j} $与 $ \sum\limits_{j=0}^{k mod p} $ \(*\) $\binom{n%p}{j} $ 则可以通过暴力预处理出

(因为模数较小)

然后 我们熟悉的 $ \binom{n/p}{i/p} $ 直接用卢卡斯就好了

代码

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. #define re register
  4. #define int long long
  5. #define in inline
  6. #define get getchar()
  7. in int read()
  8. {
  9. int t=0; char ch=get;
  10. while(ch<'0' || ch>'9') ch=get;
  11. while(ch<='9' && ch>='0')t=t*10+ch-'0', ch=get;
  12. return t;
  13. }
  14. const int mod=2333;
  15. const int _=3005;
  16. int C[_][_],sum[_][_];
  17. in int lucas(int n,int m)
  18. {
  19. if(!m||n==m) return 1;
  20. if(n<m) return 0;
  21. return lucas(n/mod,m/mod)*C[n%mod][m%mod]%mod;
  22. }
  23. in int work(int n,int k)
  24. {
  25. if(k<0)
  26. return 0;
  27. if(!n||!k)
  28. return 1;
  29. if(n<=mod&&k<=mod) return sum[n][k];
  30. return (work(n/mod,k/mod-1)*sum[n%mod][mod-1]%mod+lucas(n/mod,k/mod)*sum[n%mod][k%mod]%mod)%mod;
  31. }
  32. signed main()
  33. {
  34. int T=read();
  35. sum[0][0]=C[0][0]=1;
  36. for(re int i=1;i<=mod+3;i++)
  37. sum[i][0]=C[i][0]=C[i][i]=1;
  38. for(re int i=1;i<=mod+3;i++)
  39. for(re int j=1;j<i;j++)
  40. C[i][j]=(C[i-1][j]+C[i-1][j-1])%mod; //预处理组合数(杨辉三角)
  41. for(re int i=0;i<=mod+3;i++)
  42. for(re int j=1;j<=mod+3;j++)
  43. sum[i][j]=(C[i][j]+sum[i][j-1])%mod; //预处理杨辉三角同一行的前缀和
  44. while(T--)
  45. {
  46. int n=read(),k=read();
  47. cout<<work(n,k)<<endl;
  48. }
  49. return 0;
  50. }

Lucas(卢卡斯)定理模板&&例题解析([SHOI2015]超能粒子炮·改)的更多相关文章

  1. Bzoj 4591: [Shoi2015]超能粒子炮·改 数论,Lucas定理,排列组合

    4591: [Shoi2015]超能粒子炮·改 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 178  Solved: 70[Submit][Stat ...

  2. bzoj 4591: [Shoi2015]超能粒子炮·改 [lucas定理]

    4591: [Shoi2015]超能粒子炮·改 题意:多组询问,求 \[ S(n, k) = \sum_{i=0}^n \binom{n}{i} \mod 2333,\ k \le n \le 10^ ...

  3. 【BZOJ4591】[SHOI2015]超能粒子炮·改 (卢卡斯定理)

    [BZOJ4591][SHOI2015]超能粒子炮·改 (卢卡斯定理) 题面 BZOJ 洛谷 题解 感天动地!终于不是拓展卢卡斯了!我看到了一个模数,它是质数!!! 看着这个东西就感觉可以递归处理. ...

  4. BZOJ_4591_[Shoi2015]超能粒子炮·改_Lucas定理

    BZOJ_4591_[Shoi2015]超能粒子炮·改_Lucas定理 Description 曾经发明了脑洞治疗仪&超能粒子炮的发明家SHTSC又公开了他的新发明:超能粒子炮·改--一种可以 ...

  5. bzoj4591 / P4345 [SHOI2015]超能粒子炮·改

    P4345 [SHOI2015]超能粒子炮·改 题意:求$\sum_{i=1}^{k}C(n,i)\%(P=2333)$ 肯定要先拆开,不然怎么做呢(大雾) 把$C(n,i)$用$lucas$分解一下 ...

  6. 洛谷 P4345 [SHOI2015]超能粒子炮·改 解题报告

    P4345 [SHOI2015]超能粒子炮·改 题意 求\(\sum_{i=0}^k\binom{n}{i}\),\(T\)组数据 范围 \(T\le 10^5,n,j\le 10^{18}\) 设\ ...

  7. 【bzoj4591】[Shoi2015]超能粒子炮·改 Lucas定理

    题目描述 曾经发明了脑洞治疗仪&超能粒子炮的发明家SHTSC又公开了他的新发明:超能粒子炮·改--一种可以发射威力更加强大的粒子流的神秘装置.超能粒子炮·改相比超能粒子炮,在威力上有了本质的提 ...

  8. [bzoj4591][Shoi2015][超能粒子炮·改] (lucas定理+组合计数)

    Description 曾经发明了脑洞治疗仪&超能粒子炮的发明家SHTSC又公开了他的新发明:超能粒子炮·改--一种可以发射威力更加 强大的粒子流的神秘装置.超能粒子炮·改相比超能粒子炮,在威 ...

  9. P4345 [SHOI2015]超能粒子炮·改 Lucas

    \(\color{#0066ff}{ 题目描述 }\) 曾经发明了脑洞治疗仪与超能粒子炮的发明家 SHTSC 又公开了他的新发明:超能粒子炮・改--一种可以发射威力更加强大的粒子流的神秘装置. 超能粒 ...

随机推荐

  1. 前端性能测试工具之PageSpeed Insights

    谷歌开发的一个免费的网页分析工具,在地址栏中输入被分析的网站 url 地址,点击分析 地址:https://developers.google.cn/speed/pagespeed/insights/ ...

  2. nginx安装步骤

    1.下载地址:下载nginx压缩包wget -c https://nginx.org/download/nginx-1.10.1.tar.gz2.配置nginx安装所需的环境yum install g ...

  3. 实用向—总结一些唯一ID生成方式

    在日常的项目开发中,我们经常会遇到需要生成唯一ID的业务场景,不同的业务对唯一ID的生成方式与要求都会不尽相同,一是生成方式多种多样,如UUID.雪花算法.数据库递增等:其次业务要求上也各有不同,有的 ...

  4. Flex、Grid、媒体查询实现响应式布局

    本篇文章主要讲述使用Flex布局.Grid布局以及媒体查询三种方式来实现响应式布局. 文章涉及代码在线coding地址 效果图: 文字描述: 屏幕大小不同,展示列数不同,1-5号按照屏幕大小可展示2到 ...

  5. react 中发布订阅模式使用

    react 中发布订阅模式使用 场景 怎么能将设计模式应用到我们的 React 项目中?以前一直在思考这个问题. 场景一 模块 A 模块 B 需要用到同一个数据 data,A 和 B 都会修改这份数据 ...

  6. 正交矩阵(Orthogonal Matrix)

  7. LeetCode刷题总结-数学篇

    本文总结LeetCode上有数学类的算法题,推荐刷题总数为40道.具体考点分析如下图: 1.基本运算问题 题号:29. 两数相除,难度中等 题号:166. 分数到小数,难度中等 题号:372. 超级次 ...

  8. C++里的程序 GetDlgItem(IDC_EDIT_INPUTFILE) ->EnableWindow(TRUE)

    转载:https://zhidao.baidu.com/question/654519209423407765.html GetDlgItem(IDC_EDIT_INPUTFILE) ->Ena ...

  9. ASP。NET MVC (NetCore 2.0)用于处理实体框架、DbContexts和对象的通用控制器和视图

    下载source - 1.5 MB 介绍 本文的源代码已更新到NetCore 2.0 ASP.净MVC项目. 当我们开始开发一个ASP.在Microsoft Visual Studio中,我们发现通过 ...

  10. 获取url中带的参数

    本文目前只针对url中一个参数的 function getQueryString(name) { var reg = new RegExp("(^|&)" + name + ...