当前位置: 首页 > article >正文

快速求解组合数

欢迎访问我的个人博客:http://www.ghost-him.com/ 文章优先发布到个人博客上。

组合是数学的重要概念之一。从 n n n 个不同元素中每次取出 m m m 个不同元素 ( 0 ⩽ m ⩽ n ) (0 \leqslant m \leqslant n) (0mn) ,不管其顺序合成一组,称为从 n n n 个元素中不重复地选取 m m m 个元素的一个组合。所有这样的组合的种数称为组合数。

本文将通过动态规划的思维和数学方法来求解组合数。

方法一:动态规划

适用数据范围:c[i][j]i < 2000 且 j < 2000

通过公式
C n m = C n − 1 m − 1 + C n − 1 m C_{n}^{m} = C_{n-1}^{m-1} + C_{n-1}^{m} Cnm=Cn1m1+Cn1m
可知,求一个组合数时,只需要知道之前的两个组合数,再通过相加即可得出。

我们要理解这个公式,可以既可以通过数学公式推导,可以通过算法中动态规划的思维来理解。在高中的时候,老师已经讲过这个公式的数学推导了,所以这里就不赘述了。

这里讲一下动态规划的思维。

这个动态规划的思维是01 背包模型。我们设变量 C[i][j] 表示的是第下标是 i,上标是 j 的组合数,即 C i j C_{i}^{j} Cij
。即,这个变量表示的是在 i 个物品中选 j 个物品的方案数。我们在选第 j 个物品时,有两种方案:

  1. 第一种是第 j 个物品。
  2. 第二种是不选第 j 个物品。

我们先看第一种情况:如果选了第 j 个物品,那么当前的状态就是在 i 个物品中选了 j 个物品。同时,由于这个状态是从上一个状态转移过来的。只有当上一个状态是选了 j - 1 个物品时,加上当前的物品,才可以是当前的这个状态。因此,上一个状态就是在 i - 1 个物品中选了 j - 1 个物品的方案,即 c[i - 1][j - 1]

我们再来看第二种情况:如果我们不选这个物品。那就说明当前的状态选的物品的个数和上一个状态选的物品的个数一致。因此,上一个状态就是在 i - 1 个物品中选 j 个物品的方案,即 c[i - 1][j]

由于求的是方案的个数(组合数的定义),所以当前的状态等于转移的状态之和。即 c[i][j] = c[i - 1][j - 1] + c[i - 1][j]

题目链接:AcWing 885. 求组合数 I

我们来看一下代码的写法

#include <iostream>
#include <cstring>

using namespace std;

const int N = 2010, mod = 1e9 + 7;

// 询问的个数
int n;
// c[i][j] 表示的值
int c[N][N];

int main () {
    // 根据题目中询问的数据的范围,预处理一下c[i][j]的值
	for (int i = 0; i <= 2000; i ++) {
		for (int j = 0; j <= i; j ++) {
		    // c[i][0] = 1
			if (j == 0) c[i][j] = 1;
			else {
			    // 根据公式写
				c[i][j] = (c[i - 1][j] + c[i - 1][j - 1]) % mod;
			}
		}
	}
	// 读取询问的次数
	cin >> n;
	for (int i = 0; i < n; i ++) {
		// 读取每次查询的 下标a 和 上标j
		int a, b;
		scanf("%d%d", &a, &b);
		// 输出c[a][b]的值
		printf("%d\n", c[a][b]);
	}
	
	return 0;
}

方法二:数学公式

适用数据范围:c[i][j]i < 10000 且 j < 10000

通过公式
C n m = P n m P n = n ! m ! ( n − m ) ! C_{n}^{m} = \frac{P_{n}^{m}}{P_{n}}=\frac{n!}{m!(n-m)!} Cnm=PnPnm=m!(nm)!n!
可知如果想要求一个组合数,只需要通过阶乘运算就可以得出。

当我们要算的数字很小的时候,我们可以通过直接除法的方式来直接求出一个组合数。但是,在取余的环境下,我们不可以直接用除法。此时,如果我们需要乘以一个数,我们可以乘以这个数的逆元

假设 A A A 的逆元是 a a a,那么
C A m o d    p ≡ C ∗ a m o d    p \frac{C}{A} \mod p \equiv C*a \mod p ACmodpCamodp
通过逆元,我们可以将除法变成乘法

设一个 fact[i] 表示 i ! i! i!infact[i] 表示 i ! i! i! 的逆元;c[i][j] 表示 C i j C_{i}^{j} Cij。那么 c[i][j] = fact[i] * infact[j] * fact[n - m]

在这里,如何计算一个逆元的值呢?

通过费马小定理,我们可以知道,如果 p 是一个质数,而整数 a 不是 p 的倍数,那么
a p − 1 ≡ 1 m o d    p a^{p-1} \equiv 1\mod p ap11modp

因此,
C A m o d    p ≡ C ∗ a m o d    p \frac{C}{A} \mod p \equiv C*a \mod p ACmodpCamodp
1 ≡ a ∗ A m o d    p 1 \equiv a * A \mod p 1aAmodp
A p − 1 ≡ a ∗ A m o d    p A^{p - 1} \equiv a*A \mod p Ap1aAmodp
A p − 2 ≡ a m o d    p A^{p-2} \equiv a \mod p Ap2amodp
所以,当 p 和 A 互质的时候,A 的逆元就是 A p − 2 A^{p-2} Ap2

我们来看代码的写法。

题目:AcWing 886. 求组合数 II

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 100010, mod = 1e9 + 7;

// 要用long long 类型,所以为了少写字母,将long long 定义成LL
typedef long long LL;
// i! 的值和 i! 的逆元
int fact[N], infact[N];

// 快速幂模板,计算A的逆元
LL qmi(LL a, LL b, LL q) {
	LL res = 1;
	while (b) {
		if (b & 1) {
			res = res * a % q;
		}
		a = a * a % mod;
		b >>= 1;
	}
	return res;
}

int main () {
    // 初始化, 0! = 1, 0!的逆元也是1
	fact[0] = infact[0] = 1;
	// 求i!的值和其逆元。
	for (int i = 1; i < N; i ++) {
	    // 根据 (i - 1)! 计算 i! 的值
		fact[i] = (LL) fact[i - 1] * i % mod;
		// 根据 (i - 1)! 逆元的值计算 i! 逆元的值
		infact[i] = (LL)infact[i - 1] * qmi(i, mod - 2, mod) % mod;
	}
	
	int n;
	scanf("%d", &n);
	while (n --) {
		int a, b;
		scanf("%d%d", &a, &b);
		// 输出公式
		printf("%d\n", (LL)fact[a] * infact[b] % mod * infact[a - b] % mod);
	}
	return 0;
}

http://www.kler.cn/a/3534.html

相关文章:

  • 阿九的python 爬虫进阶课18.3 学习笔记
  • linux-ubuntu学习笔记碎记
  • QT调用OpenSceneGraph
  • 为AI聊天工具添加一个知识系统 之56 前端工具:知识图谱、语义网络和认知地图 之1
  • 使用Websocket进行前后端实时通信
  • AI需要的基础数学知识
  • 【微信小程序】-- 页面导航 -- 编程式导航(二十三)
  • chartgpt 告诉我的,loss 函数的各种知识
  • Android DataBinding 自定义View实现数据双向绑定
  • 中国蚁剑AntSword实战
  • 【Linux】Linux下权限的理解
  • 大公司为什么禁止SpringBoot项目用Tomcat?
  • 学习typeScript(weakMap,weakSet,set,map)
  • 动态规划---线性dp和区间dp
  • STM32外设-定时器详解
  • QT之QSysInfo(查看电脑信息)
  • 【springcloud 微服务】Spring Cloud Alibaba Nacos使用详解
  • 如何成为优秀的程序员
  • 并发粗略测算
  • 6.3 归并排序Mergesort
  • 【深度强化学习】(3) Policy Gradients 模型解析,附Pytorch完整代码
  • 51单片机8*8 LED点阵实现原理讲解
  • echarts地图不同地区设置不同的颜色
  • 手机验证发送及其验证(基于springboot+redis)保姆级
  • Docker【基本使用】
  • 你还不会递归?告别困惑,我来教你