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

组合数:从基础理论到高效算法实现

文章目录

  • 一、组合数学基础
  • 二、经典算法实现
  • 三、取模运算与高效算法
  • 四、算法选择策略
  • 五、典型应用场景
  • 六、进阶技巧
  • 七、常用模板
  • 总结:

一、组合数学基础

1.1 组合数定义
在离散数学中,组合数 C ( n , k ) C(n,k) C(n,k)(记为 ( n k ) \dbinom{n}{k} (kn))表示在 n n n 个不同元素中选取 k k k 个元素的非重复组合方式数量,其经典定义为:

( n k ) = n ! k ! ( n − k ) ! ( 0 ⩽ k ⩽ n ) \dbinom{n}{k} = \frac{n!}{k!(n-k)!} \quad (0 \leqslant k \leqslant n) (kn)=k!(nk)!n!(0kn)

1.2 递推关系
式如《算法导论》中常见的手法,我们从递归方程入手分析。组合数满足著名Pascal递推公式:

( n k ) = ( n − 1 k − 1 ) + ( n − 1 k ) \dbinom{n}{k} = \dbinom{n-1}{k-1} + \dbinom{n-1}{k} (kn)=(k1n1)+(kn1)

边界条件:
( n 0 ) = ( n n ) = 1 ∀ n ⩾ 0 \dbinom{n}{0} = \dbinom{n}{n} = 1 \quad \forall n \geqslant 0 (0n)=(nn)=1n0

二、经典算法实现

2.1 直接递归算法

int comb_recursive(int n, int k) {if (k == 0 || k == n) return 1;return comb_recursive(n-1, k-1) + comb_recursive(n-1, k);
}

复杂度分析:

  • 时间复杂度: O ( 2 n ) O(2^n) O(2n)(这显然无法接受)
  • 空间复杂度: O ( n ) O(n) O(n)(递归栈深度)

该算法适用于教学演示递归关系,但通过算法导论中的递归树分析可知存在大量重复计算,不具备实用价值。

2.2 动态规划解法

int comb_dp(int n, int k) {vector<vector<int>> dp(n+1, vector<int>(k+1, 0));for(int i = 0; i <= n; ++i) {for(int j = 0; j <= min(i, k); ++j) {dp[i][j] = (j == 0 || j == i) ? 1 : dp[i-1][j-1] + dp[i-1][j];}}return dp[n][k];
}

优化空间复杂度:

int comb_dp_optimized(int n, int k) {vector<int> dp(k+1, 0);dp[0] = 1;for(int i = 1; i <= n; ++i) {for(int j = min(i, k); j > 0; --j) {dp[j] += dp[j-1];}}return dp[k];
}

复杂度分析:

  • 时间复杂度: O ( n k ) O(nk) O(nk)
  • 空间复杂度:原算法 O ( n k ) O(nk) O(nk),优化后 O ( k ) O(k) O(k)

该方法通过递推存储中间结果避免了重复计算,达到多项式时间复杂度,适用于中小规模数据。

三、取模运算与高效算法

3.1 模运算预处理
当结果需要取模时( C ( n , k ) m o d p C(n,k) \bmod p C(n,k)modp),我们结合数论中的Fermat小定理实现模逆运算:

const int MOD = 1e9+7;
vector<long long> fact, inv_fact;void precompute(int n) {fact.resize(n+1);inv_fact.resize(n+1);fact[0] = 1;for(int i=1; i<=n; ++i) fact[i] = fact[i-1] * i % MOD;inv_fact[n] = mod_pow(fact[n], MOD-2);for(int i=n-1; i>=0; --i)inv_fact[i] = inv_fact[i+1] * (i+1) % MOD;
}int comb_mod(int n, int k) {return fact[n] * inv_fact[k] % MOD * inv_fact[n-k] % MOD;
}

复杂度分析:

  • 预处理时间复杂度: O ( n ) O(n) O(n)
  • 查询时间复杂度: O ( 1 ) O(1) O(1)

3.2 Lucas定理应用
对于大质数 p p p,当 n > p n > p n>p时需要采用分治策略:

int lucas_theorem(int n, int k, int p) {if(k == 0) return 1;return lucas_theorem(n/p, k/p, p) * comb_mod(n%p, k%p) % p;
}

该算法时间复杂度为 O ( log ⁡ p n ) O(\log_p n) O(logpn),主要应用于: p p p为质数且 n n n极大时的组合数取模计算。

四、算法选择策略

算法类型适用场景时间复杂度空间复杂度
Direct Recursive教学演示 O ( 2 n ) O(2^n) O(2n) O ( n ) O(n) O(n)
Dynamic Programming n ≤ 5000 n \leq 5000 n5000 O ( n 2 ) O(n^2) O(n2) O ( n ) O(n) O(n)
Modular Preprocess n ≤ 1 0 6 n \leq 10^6 n106 O ( n ) O(n) O(n) preprocessing O ( n ) O(n) O(n)
Lucas + Preprocess n ≤ 1 0 18 n \leq 10^{18} n1018, prime p ≤ 1 0 6 p \leq 10^6 p106 O ( p + log ⁡ p n ) O(p + \log_p n) O(p+logpn) O ( p ) O(p) O(p)

五、典型应用场景

5.1 二项式定理展开
( a + b ) n = ∑ k = 0 n ( n k ) a n − k b k (a+b)^n = \sum_{k=0}^{n} \dbinom{n}{k} a^{n-k}b^k (a+b)n=k=0n(kn)ankbk

5.2 组合数路径计数
m × n m \times n m×n网格图从左上到右下的路径数等于 ( m + n m ) \dbinom{m+n}{m} (mm+n)

int uniquePaths(int m, int n) {return comb_mod(m+n-2, min(m-1, n-1));
}

5.3 概率计算应用
在n次独立重复试验中的应用:
P ( k ) = ( n k ) p k ( 1 − p ) n − k P(k) = \dbinom{n}{k} p^k (1-p)^{n-k} P(k)=(kn)pk(1p)nk

六、进阶技巧

6.1 Catalan数计算
使用组合数表达式:
C n = 1 n + 1 ( 2 n n ) C_n = \frac{1}{n+1}\dbinom{2n}{n} Cn=n+11(n2n)

6.2 组合数奇偶性判定
利用Lucas定理可得:
( n k ) \dbinom{n}{k} (kn)为奇数的充要条件是 k k k的二进制表示是 n n n的子集


七、常用模板

#include <bits/stdc++.h>
using namespace std;const int MOD = 1e9 + 7;    // 模数,根据题目要求修改为质数
const int N = 1e5 + 10;     // 预处理的最大范围,根据题目要求调整long long fact[N];          // 存储阶乘 % MOD 的结果
long long inv_fact[N];      // 存储阶乘逆元 % MOD 的结果long long qpow(long long a, long long b) {long long res = 1;while (b) {if (b & 1) res = res * a % MOD;a = a * a % MOD;b >>= 1;}return res;
}
void init_comb() {fact[0] = 1;// 计算阶乘for (int i = 1; i < N; ++i) {fact[i] = fact[i - 1] * i % MOD;}// 计算最大位置阶乘的逆元inv_fact[N - 1] = qpow(fact[N - 1], MOD - 2);// 递推计算阶乘逆元for (int i = N - 2; i >= 0; --i) {inv_fact[i] = inv_fact[i + 1] * (i + 1) % MOD;}
}long long comb(int n, int k) {if (n < 0 || k < 0 || k > n) return 0;return fact[n] * inv_fact[k] % MOD * inv_fact[n - k] % MOD;
}int main() {init_comb(); // 初始化,必须调用cout << "C(5, 2) = " << comb(5, 2) << endl; // 输出 10cout << "C(100000, 50000) = " << comb(100000, 50000) << endl; // 需要保证 N > 100000return 0;
}
输出样例:
C(5, 2) = 10
C(100000, 50000) = 149033233

总结:

组合数的计算既是经典的数学问题,也蕴含着丰富的算法思想。在算法竞赛中也常常出现,要好好理解组合数的思想,感受下组合数的魅力。

相关文章:

组合数:从基础理论到高效算法实现

文章目录 一、组合数学基础二、经典算法实现三、取模运算与高效算法四、算法选择策略五、典型应用场景六、进阶技巧七、常用模板总结&#xff1a; 一、组合数学基础 1.1 组合数定义 在离散数学中&#xff0c;组合数 C ( n , k ) C(n,k) C(n,k)&#xff08;记为 ( n k ) \dbi…...

【C++】B2124 判断字符串是否为回文

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述输入格式&#xff1a;输出格式&#xff1a;样例&#xff1a; &#x1f4af;方法一&#xff1a;我的第一种做法思路代码实现解析 &#x1f4af;方法二&#xff1a;我…...

DeepSpeed Zero 解读

目录 主要参考: 分布式训练基础 – 数据并行&#xff0c;模型并行&#xff0c;流水线并行 DeepSpeed Zero 的各个 stage 介绍 针对Zero 的各个stage&#xff0c;这里有三个点需要额外再说一下&#xff1a; 各个stage&#xff0c;要实现将某一部分参数分配到不同GPU&#xff0c…...

Windows 安装Linux子系统

文章目录 一、启用虚拟化二、安装子系统1. 查看所有官方支持的 WSL 发行版2. 安装 Ubuntu3. 安装非官方发行版(如 CentOS)三、启动和更新子系统1. 启动Ubuntu终端2. 更新系统四、管理已安装的发行版在 Windows 的 WSL(Windows Subsystem for Linux)中,除了 Ubuntu,你还可…...

基于Spring Security 6的OAuth2 系列之八 - 授权服务器--Spring Authrization Server的基本原理

之所以想写这一系列&#xff0c;是因为之前工作过程中使用Spring Security OAuth2搭建了网关和授权服务器&#xff0c;但当时基于spring-boot 2.3.x&#xff0c;其默认的Spring Security是5.3.x。之后新项目升级到了spring-boot 3.3.0&#xff0c;结果一看Spring Security也升级…...

【LeetCode 刷题】回溯算法(5)-棋盘问题

此博客为《代码随想录》二叉树章节的学习笔记&#xff0c;主要内容为回溯算法棋盘问题相关的题目解析。 文章目录 51. N皇后37. 解数独332.重新安排行程 51. N皇后 题目链接 class Solution:def solveNQueens(self, n: int) -> List[List[str]]:board [[. for _ in rang…...

PaddleOCR 截图自动文字识别

春节假期在家无聊&#xff0c;撸了三个小工具&#xff1a;PC截图编辑/PC录屏(用于meeting录屏)/PC截屏文字识别。因为感觉这三个小工具是工作中常常需要用到的&#xff0c;github上也有很多开源的&#xff0c;不过总有点或多或少的小问题&#xff0c;不利于自己的使用。脚本的编…...

算法题(48):反转链表

审题&#xff1a; 需要我们将链表反转并返回头结点地址 思路&#xff1a; 一般在面试中&#xff0c;涉及链表的题会主要考察链表的指向改变&#xff0c;所以一般不会允许我们改变节点val值。 这里是单向链表&#xff0c;如果要把指向反过来则需要同时知道前中后三个节点&#x…...

梯度、梯度下降、最小二乘法

在求解机器学习算法的模型参数&#xff0c;即无约束优化问题时&#xff0c;梯度下降是最常采用的方法之一&#xff0c;另一种常用的方法是最小二乘法。 1. 梯度和梯度下降 在微积分里面&#xff0c;对多元函数的参数求∂偏导数&#xff0c;把求得的各个参数的偏导数以向量的形式…...

独立开发者小程序开发变现思路

随着移动互联网的发展&#xff0c;小程序已成为许多独立开发者展示才能和实现收入的重要平台。作为一种轻量级的应用形态&#xff0c;小程序具有开发成本低、用户体验好、传播效率高等优势&#xff0c;为独立开发者提供了多种变现方式。然而&#xff0c;要想实现真正的盈利&…...

软件测试 - 概念篇

目录 1. 需求 1.1 用户需求 1.2 软件需求 2. 开发模型 2.1 软件的生命周期 2.2 常见开发模型 2.2.1 瀑布模型 2.2.2 螺旋模型 1. 需求 对于软件开发而言, 需求分为以下两种: 用户需求软件需求 1.1 用户需求 用户需求, 就是用户提出的需求, 没有经过合理的评估, 通常…...

使用SpringBoot发送邮件|解决了部署时连接超时的bug|网易163|2025

使用SpringBoot发送邮件 文章目录 使用SpringBoot发送邮件1. 获取网易邮箱服务的授权码2. 初始化项目maven部分web部分 3. 发送邮件填写配置EmailSendService [已解决]部署时连接超时附&#xff1a;Docker脚本Dockerfile创建镜像启动容器 1. 获取网易邮箱服务的授权码 温馨提示…...

基于springboot+vue的航空散货调度系统

开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#xff1a;…...

自学习记录-编程语言的特点(持续记录)

我学习的顺序是C -> python -> C -> Java。在讲到某项语言的特点是&#xff0c;可能会时不时穿插其他语言的特点。 Java 1 注解Annotation Python中也有类似的Decorators。以下为AI学习了解到的&#xff1a; Java的Annotation是一种元数据&#xff08;metadata)&a…...

[MRCTF2020]Ez_bypass1(md5绕过)

[MRCTF2020]Ez_bypass1(md5绕过) ​​ 这道题就是要绕过md5强类型比较&#xff0c;但是本身又不相等&#xff1a; md5无法处理数组&#xff0c;如果传入的是数组进行md5加密&#xff0c;会直接放回NULL&#xff0c;两个NuLL相比较会等于true&#xff1b; 所以?id[]1&gg…...

MATLAB实现多种群遗传算法

多种群遗传算法&#xff08;MPGA, Multi-Population Genetic Algorithm&#xff09;是一种改进的遗传算法&#xff0c;它通过将种群分成多个子种群并在不同的子种群之间进行交叉和交换&#xff0c;旨在提高全局搜索能力并避免早期收敛。下面是多种群遗传算法的主要步骤和流程&a…...

强化学习笔记(5)——PPO

PPO视频课程来源 首先理解采样期望的转换 变量x在p(x)分布下&#xff0c;函数f(x)的期望 等于f(x)乘以对应出现概率p(x)的累加 经过转换后变成 x在q(x)分布下&#xff0c;f(x)*p(x)/q(x) 的期望。 起因是&#xff1a;求最大化回报的期望&#xff0c;所以对ceta求梯度 具体举例…...

【MATLAB例程】TOA和AOA混合的高精度定位程序,适用于三维、N锚点的情况

代码实现了一个基于到达角&#xff08;AOA&#xff09;和到达时间&#xff08;TOA&#xff09;混合定位的例程。该算法能够根据不同基站接收到的信号信息&#xff0c;自适应地计算目标的位置&#xff0c;适用于多个基站的场景 文章目录 主要功能代码结构运行结果程序代码 主要功…...

使用Pygame制作“青蛙过河”游戏

本篇博客将演示如何使用 Python Pygame 从零开始编写一款 Frogger 风格的小游戏。Frogger 是一款早期街机经典&#xff0c;玩家需要帮助青蛙穿越车水马龙的马路到达对岸。本示例提供了一个精简原型&#xff0c;包含角色移动、汽车生成与移动、碰撞检测、胜利条件等关键点。希望…...

深度解读 Docker Swarm

一、引言 随着业务规模的不断扩大和应用复杂度的增加,容器集群管理的需求应运而生。如何有效地管理和调度大量的容器,确保应用的高可用性、弹性伸缩和资源的合理分配,成为了亟待解决的问题。Docker Swarm 作为 Docker 官方推出的容器集群管理工具,正是在这样的背景下崭露头…...

8、面向对象:类、封装、构造方法

一、类 1、定义 类&#xff1a;对现实世界中事物的抽象。Student 对象&#xff1a;现实世界中具体的个体。张三、李四 这些具体的学生 面向对象的特征&#xff1a;抽象、封装、继承、多态 OOP: Object Oriented Programming&#xff08;面向对象编程&#xff09; 类和对象…...

AI时代IT行业职业方向规划大纲

一、引言 AI时代的颠覆性影响 ChatGPT、Midjourney等生成式AI对传统工作模式的冲击 案例&#xff1a;AI编程助手&#xff08;GitHub Copilot&#xff09;改变开发者工作流程 核心问题&#xff1a;IT从业者如何避免被AI替代&#xff0c;并找到新机遇&#xff1f; 二、AI时代…...

STM32 旋转编码器

旋转编码器简介 旋转编码器&#xff1a;用来测量位置、速度或旋转方向的装置&#xff0c;当其旋转轴旋转时&#xff0c;其输出端可以输出与旋转速度和方向对应的方波信号&#xff0c;读取方波信号的频率和相位信息即可得知旋转轴的速度和方向 类型&#xff1a;机械触点式/霍尔传…...

大模型领域的Scaling Law的含义及作用

Scaling Law就像是一个“长大公式”&#xff0c;用来预测当一个东西&#xff08;比如模型&#xff09;变大&#xff08;比如增加参数、数据量&#xff09;时&#xff0c;它的性能&#xff08;比如准确率&#xff09;会怎么变化。 它能帮助我们提前知道&#xff0c;增加多少资源…...

从 C 到 C++:理解结构体中字符串的存储与操作

对于刚入门 C/C 的程序员来说&#xff0c;字符串的存储和操作可能是个容易混淆的知识点。在 C 中&#xff0c;std::string 提供了非常友好的接口&#xff0c;我们可以轻松地在结构体中使用字符串类型&#xff0c;无需关注底层细节。然而&#xff0c;在 C 语言中&#xff0c;字符…...

git基础使用--4---git分支和使用

文章目录 git基础使用--4---git分支和使用1. 按顺序看2. 什么是分支3. 分支的基本操作4. 分支的基本操作4.1 查看分支4.2 创建分支4.3 切换分支4.4 合并冲突 git基础使用–4—git分支和使用 1. 按顺序看 -git基础使用–1–版本控制的基本概念 -git基础使用–2–gti的基本概念…...

【算法】回溯算法专题③ ——排列型回溯 python

目录 前置小试牛刀回归经典举一反三总结 前置 【算法】回溯算法专题① ——子集型回溯 python 【算法】回溯算法专题② ——组合型回溯 剪枝 python 小试牛刀 全排列 https://leetcode.cn/problems/permutations/description/ 给定一个不含重复数字的数组 nums &#xff0c;返…...

Vue2.x简介

Vue2.x简介 Vue2.x的版本介绍Vue2.x的两大组件库 Vue2.x的版本介绍 Vue2.x是vue.js的第二个主要版本&#xff0c;最初版发布于2016 年&#xff0c;最终版发布于2023年12月24日&#xff08;版本号&#xff1a;2.7.16&#xff0c;版本名&#xff1a;Swan Song&#xff08;绝唱&a…...

FFmpeg:多媒体处理的瑞士军刀

FFmpeg&#xff1a;多媒体处理的瑞士军刀 前言 FFmpeg 是一个功能强大且跨平台的开源多媒体框架&#xff0c;广泛应用于音视频处理领域。 它由多个库和工具组成&#xff0c;能够处理各种音视频格式&#xff0c;涵盖编码、解码、转码、流处理等多种操作。 无论是专业视频编辑…...

Windows编译FreeRDP步骤

1. **安装必要工具** powershell # 安装 Visual Studio 2022 (勾选"C桌面开发"组件) # 安装 CMake: https://cmake.org/download/ # 安装 Git: https://git-scm.com/ 2. **安装依赖项** powershell # 使用vcpkg包管理 git clone https://github.com/Microsoft/vcpk…...