一个有趣的问题

题目大意

  给定$n, m, p$,求$C_{n}^{m}$除以$p$后的余数。

Subtask#1  $0\leqslant m\leqslant n \leqslant 2\times 10^{3}$

  直接杨辉恒等式$C_{n}^{m} = C_{n - 1}^{m - 1} + C_{n - 1}^{m}$递推。

  时间复杂度$O(n^{2})$。

Subtask#2  $0\leqslant m\leqslant n \leqslant 2\times 10^{6},p = 10^{9} + 7$

  用式子$C_{n}^{m} = \frac{n!}{m!(n - m)!}$来计算。

  直接算阶乘,用快速幂或扩欧求逆元即可。

  时间复杂度$O(n + \log n)$

Subtask#3 $0\leqslant m\leqslant n \leqslant 10^{9},p\leqslant 10^{6}$且$p$为质数。

  好吧,现在引入正题。

定理1(Lucas定理) 当$p$是一个质数时,$n = k_{1}p + r_{1}, m = k_{2}p + r_{2} \left ( 0\leqslant r_{1},r_{2} < p \right )$,则有$C_{n}^{m} \equiv C_{k_{1}}^{k_{2}}C_{r_{1}}^{r_{2}} \pmod{p}$

  考虑怎么使用它。

  对于每次操作的$r_{1}$和$r_{2}$都是小于$p$的,这个可以直接用Subtask 2的方法来做。

  然后剩下的$C_{k_{1}}^{k_{2}}$就递归处理。

  这很简单,就不给代码了。

  现在来考虑Lucas定理的证明。

  首先你需要一个二项式定理。

定理2(二项式定理) 当$n$是一个非负整数时,则有,$\left (a + b  \right ) ^{n} = \sum_{i = 0} ^{n}C_{n}^{i}a^{i}b^{n - i}$

  证明 考虑使用数学归纳法来证明。

  当$n = 1$的时候,显然成立。

  假设当$n = k - 1$的时候成立,考虑当$n = k$的时候

$\left (a + b  \right )^{k} = \left ( a + b \right )^{k - 1}\left ( a + b \right )\\=\left ( \sum_{i = 0}^{k - 1}C_{k-1}^{i}a^{i}b^{k - i - 1} \right )(a + b)\\=\sum_{i = 1}^{k}C_{k - 1}^{i - 1}a^{i}b^{k - i} + \sum_{i = 0}^{k - 1}C_{k - 1}^{i}a^{i}b^{k - i}\\=C_{k - 1}^{k - 1}a^{n} + \sum_{i = 1}^{k - 1}\left ( C_{k - 1}^{i - 1} + C_{k - 1}^{i} \right )a^{i}b^{k - i} + C_{k - 1}^{0}b^{n}\\=\sum_{i = 0}^{k} C_{k}^{i} a^{i}b^{k - i}$

  因此,定理得证。

定理3 若$p$为质数,若有整数$k$满足$1\leqslant k < p$,则有$p \mid C_{p}^{k}$

  根据式子$C_{p}^{k} = \frac{p!}{k!\left( p - k \right)!}$和素数的定义易证。

推论4 若$p$为质数,则$\left ( 1 + x\right )^{p}\equiv 1 + x^{p} \pmod{p}$。

  根据定理3和二项式定理易证。

  现在来考虑Lucas定理的证明。

  Lucas定理的证明

$\left ( 1 + x\right )^{n}\\\equiv \left ( 1 + x \right )^{k_{1}p}\left( 1 + x\right )^{r_{1}} \\\equiv \left ( 1 + x^{p} \right )^{k_{1}}\left( 1 + x\right )^{r_{1}}\\\equiv \left ( \sum_{i = 0}^{k_{1}}C_{k_{1}}^{i}x^{pi} \right )\left (   \sum_{i = 0}^{r_{1}}C_{r_{1}}^{i}x^{i} \right )\\\equiv \sum_{i = 0}^{k_{1}}\sum_{j = 0}^{r_{1}}C_{k_{1}}^{i}C_{r_{1}}^{j}x^{pi + j} \pmod{p}$

  然后考虑$x^{m}$的系数,同余式左边显然是$C_{n}^{m}$,右边是$C_{k_{1}}^{k_{2}}C_{r_{1}}^{r_{2}}$,因此定理得证。

扩展Lucas算法

  继续回到上面的问题。

Subtask#4 $0 \leqslant m \leqslant n \leqslant 10^{7},p\leqslant 10^{9}$

  继续考虑组合数的计算公式$C_{n}^{m} = \frac{n!}{m!(n - m)!}$,考虑将$n!, m!, (n - m)!$质因数分解,这样直接就可以指数相加减,最后乘起来就好了。

  考虑怎么数$n!$中质因子$p$的指数。

  我们知道$n! = 1\times 2\times 3 \times \cdots \times n$,其中是$p$的倍数的是$p, 2p, 3p, \cdots$,考虑将这些$p$除掉,然后它会变成$1, 2, 3, \cdots, \left \lfloor \frac{n}{p} \right \rfloor$。这是一个规模更小的子问题,递归处理即可。时间复杂度$O(log_{p}n)$。

  可以用线性筛先预处理$10^{7}$以内的素数,然后考虑$n!$的时候对小于$n$的每个素数去跑一次这个算法,因为$n$以内约有$\frac{n}{\log n}$个素数,所以时间复杂度为$O(n)$。

Subtask#5 $0 \leqslant m \leqslant n \leqslant 10^{9},p\leqslant 10^{9}$,若将$p$质因数分解得到:$p = p_{1}^{a_{1}}\cdots p_{k}^{a_{k}}$,,则对于任意$i = 1, 2, 3,\cdots, k$都有$p_{i}^{a_{i}}\leqslant 10^{5}$

  数据范围怎么越来越不友好了?话说后面那一大句又是什么奇怪的性质?

  继续考虑直接暴力,上面通过质因数分解的方法已经不可行。

  以上NB的方法是阶乘加逆元对吧?因为模数不一定是质数,所以和$n$之类的不一定互质,也就是说逆元可能不存在。

  设答案为$x$,那么可以将原问题拆成$k$个子问题:

$\left\{\begin{matrix}x\equiv C_{n}^{m} \pmod{p_{1}^{a_{1}}} \\ x\equiv C_{n}^{m} \pmod{p_{2}^{a_{2}}}\\ \vdots \\  x\equiv C_{n}^{m} \pmod{p_{k}^{a_{k}}}\end{matrix}\right.$

  这有什么卵用?

  假如能够把$p_{i}$抽出去单独计算,剩下就可以"快速阶乘",因为此时和模数互质,所以可以直接用逆元在模意义下做除法。最后用中国剩余定理合并。

  把$p_{i}$抽出去可以用上面的方法进行计算。

  现在考虑如何快速计算模$p_{i}^{a_{i}}$下,被抽取所有有关$p_{i}$的因子的$n!$。

  我们直接暴力枚举$1$ ~ $p_{i}^{a_{i}}$中间的数,然后暴力将不是$p_{i}$的倍数的数乘起来,对于中间在模意义下是一样的,快速幂乘一乘,最后末尾的暴力做一下。

  然后把是$p_{i}$的倍数的除掉一个$p_{i}$,这样问题的规模会变小,递归处理即可

  举个例子有助于说明:

假设$p_{i} = 3, a_{i} = 2, n = 19$,那么要计算的是:$1 \times 2\times 3 \times 4 \times 5\times 6 \times 7 \times 8 \times 9\times 10 \times 11  \times 12\times 13 \times 14 \times 15 \times 16 \times 17 \times 18 \times 19$

显然它同余于$\left (1 \times 2\times 4 \times 5 \times 7 \times 8   \right )\times \left (1 \times 2\times 4 \times 5 \times 7 \times 8   \right ) \times 1 \times 3\times \left ( 1 \times 2\times 3 \times 4 \times 5\times 6 \right )$

前面的用快速幂,末尾余下的暴力算,后面的递归处理。

  时间复杂度O(能过)

  没错,这就是扩展Lucas,我很好奇它和Lucas有什么关系。这明明就是依赖特殊性质的暴力qwq。

  最后附上代码。

Code

 /**
* bzoj
* Problem#2142
* Accepted
* Time: 372ms
* Memory: 1292k
*/
#include <bits/stdc++.h>
#ifndef WIN32
#define Auto "%lld"
#else
#define Auto "%I64d"
#endif
using namespace std;
typedef bool boolean; #define ll long long int qpow(int a, int pos, int m) {
int pa = a, rt = ;
for ( ; pos; pos >>= , pa = pa * 1ll * pa % m)
if (pos & )
rt = rt * 1ll * pa % m;
return rt;
} void exgcd(ll a, ll b, ll& d, ll& x, ll& y) {
if (!b)
d = a, x = , y = ;
else {
exgcd(b, a % b, d, y, x);
y -= (a / b) * x;
}
} ll inv(ll a, ll n) {
ll d, x, y;
exgcd(a, n, d, x, y);
return (x < ) ? (x + n) : (x);
} ll p;
int n, m, k;
int top = ;
int w[];
int fac[];
int val[]; inline void init() {
scanf(Auto"%d%d", &p, &n, &m);
ll ws = ;
for (int i = ; i <= m; i++) {
scanf("%d", w + i);
ws += w[i];
}
if (ws > n) {
puts("Impossible");
exit();
}
k = ws;
} void getFactors(ll x) {
for (int i = ; x != ; i++) {
if (!(x % i)) {
fac[++top] = i, val[top] = ;
while (!(x % i)) val[top] *= i, x /= i;
}
}
} int getPos(int n, int p) {
int rt = ;
while (n) rt += (n /= p);
return rt;
} int calc(int n, int p, int pr) {
if (!n) return ;
int rt = ;
for (int i = ; i < pr; i++)
if (i % p)
rt = rt * 1ll * i % pr;
rt = qpow(rt, n / pr, pr);
for (int i = ; i <= n % pr; i++)
if (i % p)
rt = rt * 1ll * i % pr;
return rt * 1ll * calc(n / p, p, pr) % pr;
} ll exLucas(int n, int m, ll ap) {
ll rt = , C;
for (int i = , t1, t2, t3, r1, r2, r3, p, pr; i <= top; i++) {
p = fac[i], pr = val[i];
r1 = getPos(n, p), r2 = getPos(m, p), r3 = getPos(n - m, p);
t1 = calc(n, p, pr), t2 = calc(m, p, pr), t3 = calc(n - m, p, pr);
C = ((t1 * inv(t2, pr) % pr) * inv(t3, pr) % pr * 1ll * qpow(p, r1 - r2 - r3, pr)) % pr;
rt = (rt + (((ap / pr) * inv(ap / pr, pr) % ap) * C % ap)) % ap;
}
return rt;
} inline void solve() {
getFactors(p);
ll C = exLucas(n, k, p);
for (int i = ; i <= m; i++) {
C = C * exLucas(k, w[i], p) % p;
k -= w[i];
}
printf(Auto, C);
} int main() {
init();
solve();
return ;
}

最新文章

  1. oracle 全文检索创建脚本示例
  2. Codeforces 498C Array and Operations(最大流)
  3. 浅谈C#浅拷贝和深拷贝
  4. pymol编译
  5. 转摘:常用ubuntu 关机,重启,注销命令
  6. javascript、js操作json方法总结(json字符创转换json对象)
  7. MySQL重置root用户密码的方法(转)
  8. PAC全自动脚本代理
  9. delphi中获得进程列表或想要的进程(枚举进程、遍历进程)
  10. Android热修复技术选型——三大流派解析
  11. openstack中的环境准备
  12. AJAX-同源策略 跨域访问
  13. CentOS6源码安装vim8
  14. Android横、竖屏幕动态切换(layout-land 和layout-port)
  15. [SpringBoot] - 了解什么是SpringBoot,使用SpringBoot的配置文件
  16. Java NIO学习与记录(四): SocketChannel与BIO服务器
  17. JMeter 十:录制脚本--使用bodboy
  18. 2017南宁现场赛E The Champion
  19. Android开发问题:ActivityNotFoundException: Unable to find explicit activity class
  20. Select语句完整的执行顺序

热门文章

  1. unity3d-游戏实战突出重围,第四天 添加角色
  2. mybatis多表关联查询之resultMap单个对象
  3. 加减plugin
  4. Javascript-for循环案例-打印1-100之间所有的数字
  5. 1.安装Python3和PyCharm
  6. [12]Windows内核情景分析 --- MDI
  7. 添加Google搜索
  8. Yii DataProvider
  9. 对象的copy
  10. 在lua中从一个字符串中移除空间源码