OI loves Math(三)——组合数

又来了(喜)
这次我们聊组合数,也就是 $ C_n^m $ 。。。。。。

何为组合数

没人不知道吧。。。。。。
组合数就是问从 $ n $ 项里选 $ m $ 项有多少种选法,记作 $ C_n^m $ 或 $ \left( \begin{matrix} n \\ m \end{matrix} \right) $ 。
但是,竞赛中一般求 $ C_n^m \bmod 998244353 $ 。

怎么求

注:下文 $ mod $ 为模数。
有很多,但这儿介绍一种 $ O(n \log mod) $ 预处理,$ O(1) $ 单词查询的方法!
其实就是组合数公式。。。。。。

\[C_n^m = \frac{n!}{m!(n - m)!} \]

再加上以前的逆元:

\[\frac{1}{x} \equiv x^{mod - 2} \pmod{mod} \]

预处理 $ x! $ 和 $ x!^{mod - 2} \bmod mod $ 就可以了。
两段小代码:

prod[0] = 1, inv[0] = 1;
for (long long i = 1; i <= maxn; i++) {
	prod[i] = prod[i - 1] * i % mod;
	inv[i] = qpow(prod[i], mod - 2, mod);
}
long long C(long long n, long long m) {
	return prod[n] * inv[n - m] % mod * inv[m] % mod;
}

例题

CF1717D

把选手按编号顺序排列:(注:转载于OIer某罗

我们可以发现,选手 $ x $ 进行 $ x - 1 $ ,然后 $ \operatorname{popcount}(x - 1) $ 就是胜的场数。
那么问题就变成了求满足 $ \operatorname{popcount}(x) \le k \ \ \ (0 \le x \le 2^n - 1) $ 的 $ x $ 的个数,也就是 $ \sum _ {i = 0} ^ {\min \{ n, k \}} C _ {n} ^ {i} $
代码:

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

long long prod[100005], inv[100005];
const long long mod = 1000000007;

long long qmul(long long base, long long exp, long long mod) {
	long long ans = 0;
	while (exp) {
		if (exp & 1) {
			ans = (ans + base) % mod;
		}
		exp >>= 1;
		base = (base + base) % mod;
	}
	return ans;
}

long long qpow(long long base, long long exp, long long mod) {
	long long ans = 1;
	while (exp) {
		if (exp & 1) {
//			ans = qmul(ans, base, mod);
			ans = ans * base % mod;
		}
		exp >>= 1;
//		base = qmul(base, base, mod);
		base = base * base % mod;
	}
	return ans;
}

long long C(long long n, long long m) {
	return prod[n] * inv[n - m] % mod * inv[m] % mod;
}

int main() {
	long long n, k;
	scanf("%lld %lld", &n, &k);
	long long ans = 0;
	prod[0] = 1, inv[0] = 1;
	k = min(n, k);
	for (long long i = 1; i <= n; i++) {
		prod[i] = prod[i - 1] * i % mod;
		inv[i] = qpow(prod[i], mod - 2, mod);
	}
	for (long long i = 0; i <= k; i++) {
		ans = (ans + C(n, i)) % mod;
	}
	printf("%lld", ans);
	return 0;
}

再见!

posted @ 2022-11-10 22:11  A-Problem-Solver  阅读(31)  评论(0编辑  收藏  举报