排列组合
引入
排列组合是组合数学中的基础。排列就是指从给定个数的元素中取出指定个数的元素进行排序;组合则是指从给定个数的元素中仅仅取出指定个数的元素,不考虑排序。排列组合的中心问题是研究给定要求的排列和组合可能出现的情况总数。排列组合与古典概率论关系密切。
在高中初等数学中,排列组合多是利用列表、枚举等方法解题。
加法 & 乘法原理
加法原理
完成一个工程可以有 𝑛
类办法,𝑎𝑖(1 ≤𝑖 ≤𝑛)
代表第 𝑖
类方法的数目。那么完成这件事共有 𝑆 =𝑎1 +𝑎2 +⋯ +𝑎𝑛
种不同的方法。
乘法原理
完成一个工程需要分 𝑛
个步骤,𝑎𝑖(1 ≤𝑖 ≤𝑛)
代表第 𝑖
个步骤的不同方法数目。那么完成这件事共有 𝑆 =𝑎1 ×𝑎2 ×⋯ ×𝑎𝑛
种不同的方法。
排列与组合基础
排列数
从 𝑛
个不同元素中,任取 𝑚
(𝑚 ≤𝑛
,𝑚
与 𝑛
均为自然数,下同)个元素按照一定的顺序排成一列,叫做从 𝑛
个不同元素中取出 𝑚
个元素的一个排列;从 𝑛
个不同元素中取出 𝑚
(𝑚 ≤𝑛
) 个元素的所有排列的个数,叫做从 𝑛
个不同元素中取出 𝑚
个元素的排列数,用符号 A𝑚𝑛
(或者是 P𝑚𝑛
)表示。
排列的计算公式如下:
A𝑚𝑛=𝑛(𝑛−1)(𝑛−2)⋯(𝑛−𝑚+1)=𝑛!(𝑛−𝑚)!
𝑛!
代表 𝑛
的阶乘,即 6! =1 ×2 ×3 ×4 ×5 ×6
。
公式可以这样理解:𝑛
个人选 𝑚
个来排队 (𝑚 ≤𝑛
)。第一个位置可以选 𝑛
个,第二位置可以选 𝑛 −1
个,以此类推,第 𝑚
个(最后一个)可以选 𝑛 −𝑚 +1
个,得:
A𝑚𝑛=𝑛(𝑛−1)(𝑛−2)⋯(𝑛−𝑚+1)=𝑛!(𝑛−𝑚)!
全排列:𝑛
个人全部来排队,队长为 𝑛
。第一个位置可以选 𝑛
个,第二位置可以选 𝑛 −1
个,以此类推得:
A𝑛𝑛=𝑛(𝑛−1)(𝑛−2)⋯3×2×1=𝑛!
全排列是排列数的一个特殊情况。
组合数
从 𝑛
个不同元素中,任取 𝑚 ≤𝑛
个元素组成一个集合,叫做从 𝑛
个不同元素中取出 𝑚
个元素的一个组合;从 𝑛
个不同元素中取出 𝑚 ≤𝑛
个元素的所有组合的个数,叫做从 𝑛
个不同元素中取出 𝑚
个元素的组合数,用符号 (𝑛𝑚)
来表示,读作「𝑛
选 𝑚
」。
组合数计算公式
(𝑛𝑚)=A𝑚𝑛𝑚!=𝑛!𝑚!(𝑛−𝑚)!
如何理解上述公式?我们考虑 𝑛
个人选 𝑚
个出来(𝑚 ≤𝑛
),不排队,不在乎顺序。如果在乎顺序那么就是 A𝑚𝑛
,如果不在乎那么就要除掉重复,那么重复了多少?同样选出来的 𝑚
个人,他们还要「全排」得 𝑚!
,所以得:
(𝑛𝑚)×𝑚!=A𝑚𝑛(𝑛𝑚)=A𝑚𝑛𝑚!=𝑛!𝑚!(𝑛−𝑚)!
组合数也常用 C𝑚𝑛
表示,即 C𝑚𝑛 =(𝑛𝑚)
。现在数学界普遍采用 (𝑛𝑚)
的记号而非 C𝑚𝑛
。
组合数也被称为「二项式系数」,下文二项式定理将会阐述其中的联系。
特别地,规定当 𝑚 >𝑛
时,A𝑚𝑛 =(𝑛𝑚) =0
。
插板法
插板法(Stars and bars)是用于求一类给相同元素分组的方案数的一种技巧,也可以用于求一类线性不定方程的解的组数。
正整数和的数目
问题一:现有 𝑛
个 完全相同 的元素,要求将其分为 𝑘
组,保证每组至少有一个元素,一共有多少种分法?
考虑拿 𝑘 −1
块板子插入到 𝑛
个元素两两形成的 𝑛 −1
个空里面。
因为元素是完全相同的,所以答案就是 (𝑛−1𝑘−1)
。
本质是求 𝑥1 +𝑥2 +⋯ +𝑥𝑘 =𝑛
的正整数解的组数。
非负整数和的数目
问题二:如果问题变化一下,每组允许为空呢?
显然此时没法直接插板了,因为有可能出现很多块板子插到一个空里面的情况,非常不好计算。
我们考虑创造条件转化成有限制的问题一,先借 𝑘
个元素过来,在这 𝑛 +𝑘
个元素形成的 𝑛 +𝑘 −1
个空里面插板,答案为
(𝑛+𝑘−1𝑘−1)=(𝑛+𝑘−1𝑛)
虽然不是直接求的原问题,但这个式子就是原问题的答案,可以这么理解:
开头我们借来了 𝑘
个元素,用于保证每组至少有一个元素,插完板之后再把这 𝑘
个借来的元素从 𝑘
组里面拿走。因为元素是相同的,所以转化过的情况和转化前的情况可以一一对应,答案也就是相等的。
由此可以推导出插板法的公式:(𝑛+𝑘−1𝑛)
。
本质是求 𝑥1 +𝑥2 +⋯ +𝑥𝑘 =𝑛
的非负整数解的组数(即要求 𝑥𝑖 ≥0
)。
不同下界整数和的数目
问题三:如果再扩展一步,要求对于第 𝑖
组,至少要分到 𝑎𝑖,∑𝑎𝑖 ≤𝑛
个元素呢?
本质是求 𝑥1 +𝑥2 +⋯ +𝑥𝑘 =𝑛
的解的数目,其中 𝑥𝑖 ≥𝑎𝑖
。
类比无限制的情况,我们借 ∑𝑎𝑖
个元素过来,保证第 𝑖
组至少能分到 𝑎𝑖
个。也就是令
𝑥′𝑖=𝑥𝑖−𝑎𝑖
得到新方程:
(𝑥′1+𝑎1)+(𝑥′2+𝑎2)+⋯+(𝑥′𝑘+𝑎𝑘)=𝑛𝑥′1+𝑥′2+⋯+𝑥′𝑘=𝑛−𝑎1−𝑎2−⋯−𝑎𝑘𝑥′1+𝑥′2+⋯+𝑥′𝑘=𝑛−∑𝑎𝑖
其中
𝑥′𝑖≥0
然后问题三就转化成了问题二,直接用插板法公式得到答案为
(𝑛−∑𝑎𝑖+𝑘−1𝑛−∑𝑎𝑖)
不相邻的排列
1 ∼𝑛
这 𝑛
个自然数中选 𝑘
个,这 𝑘
个数中任何两个数都不相邻的组合有 (𝑛−𝑘+1𝑘)
种。
二项式定理
在进入排列组合进阶篇之前,我们先介绍一个与组合数密切相关的定理——二项式定理。
二项式定理阐明了一个展开式的系数:
(𝑎+𝑏)𝑛=𝑛∑𝑖=0(𝑛𝑖)𝑎𝑛−𝑖𝑏𝑖
证明可以采用数学归纳法,利用 (𝑛𝑘) +(𝑛𝑘−1) =(𝑛+1𝑘)
做归纳。
二项式定理也可以很容易扩展为多项式的形式:
设 𝑛
为正整数,𝑥𝑖
为实数,
(𝑥1+𝑥2+⋯+𝑥𝑡)𝑛=∑满足𝑛1+⋯+𝑛𝑡=𝑛的非负整数解(𝑛𝑛1,𝑛2,⋯,𝑛𝑡)𝑥𝑛11𝑥𝑛22⋯𝑥𝑛𝑡𝑡
其中的 (𝑛𝑛1,𝑛2,⋯,𝑛𝑡)
是多项式系数,它的性质也很相似:
∑(𝑛𝑛1,𝑛2,⋯,𝑛𝑡)=𝑡𝑛
排列与组合进阶篇
接下来我们介绍一些排列组合的变种。
多重集的排列数 | 多重组合数
请大家一定要区分 多重组合数 与 多重集的组合数!两者是完全不同的概念!
多重集是指包含重复元素的广义集合。设 𝑆 ={𝑛1 ⋅𝑎1,𝑛2 ⋅𝑎2,⋯,𝑛𝑘 ⋅𝑎𝑘}
表示由 𝑛1
个 𝑎1
,𝑛2
个 𝑎2
,…,𝑛𝑘
个 𝑎𝑘
组成的多重集,𝑆
的全排列个数为
𝑛!∏𝑘𝑖=1𝑛𝑖!=𝑛!𝑛1!𝑛2!⋯𝑛𝑘!
相当于把相同元素的排列数除掉了。具体地,你可以认为你有 𝑘
种不一样的球,每种球的个数分别是 𝑛1,𝑛2,⋯,𝑛𝑘
,且 𝑛 =𝑛1 +𝑛2 +… +𝑛𝑘
。这 𝑛
个球的全排列数就是 多重集的排列数。多重集的排列数常被称作 多重组合数。我们可以用多重组合数的符号表示上式:
(𝑛𝑛1,𝑛2,⋯,𝑛𝑘)=𝑛!∏𝑘𝑖=1𝑛𝑖!
可以看出,(𝑛𝑚)
等价于 (𝑛𝑚,𝑛−𝑚)
,只不过后者较为繁琐,因而不采用。
多重集的组合数 1
设 𝑆 ={𝑛1 ⋅𝑎1,𝑛2 ⋅𝑎2,⋯,𝑛𝑘 ⋅𝑎𝑘}
表示由 𝑛1
个 𝑎1
,𝑛2
个 𝑎2
,…,𝑛𝑘
个 𝑎𝑘
组成的多重集。那么对于整数 𝑟(𝑟 <𝑛𝑖,∀𝑖 ∈[1,𝑘])
,从 𝑆
中选择 𝑟
个元素组成一个多重集的方案数就是 多重集的组合数。这个问题等价于 𝑥1 +𝑥2 +⋯ +𝑥𝑘 =𝑟
的非负整数解的数目,可以用插板法解决,答案为
(𝑟+𝑘−1𝑘−1)
多重集的组合数 2
考虑这个问题:设 𝑆 ={𝑛1 ⋅𝑎1,𝑛2 ⋅𝑎2,⋯,𝑛𝑘 ⋅𝑎𝑘,}
表示由 𝑛1
个 𝑎1
,𝑛2
个 𝑎2
,…,𝑛𝑘
个 𝑎𝑘
组成的多重集。那么对于正整数 𝑟
,从 𝑆
中选择 𝑟
个元素组成一个多重集的方案数。
这样就限制了每种元素的取的个数。同样的,我们可以把这个问题转化为带限制的线性方程求解:
∀𝑖∈[1,𝑘], 𝑥𝑖≤𝑛𝑖, 𝑘∑𝑖=1𝑥𝑖=𝑟![\forall i\in [1,k],\ x_i\le n_i,\ \sum_{i=1}^kx_i=r]()
于是很自然地想到了容斥原理。容斥的模型如下:
- 全集:𝑘∑𝑖=1𝑥𝑖 =𝑟
的非负整数解。 - 属性:𝑥𝑖 ≤𝑛𝑖
。
于是设满足属性 𝑖
的集合是 𝑆𝑖
,――𝑆𝑖
表示不满足属性 𝑖
的集合,即满足 𝑥𝑖 ≥𝑛𝑖 +1
的集合(转化为上面插板法的问题三)。那么答案即为
∣𝑘⋂𝑖=1𝑆𝑖∣=|𝑈|−∣𝑘⋃𝑖=1――𝑆𝑖∣
根据容斥原理,有:
∣𝑘⋃𝑖=1――𝑆𝑖∣=∑𝑖∣――𝑆𝑖∣−∑𝑖,𝑗∣――𝑆𝑖∩――𝑆𝑗∣+∑𝑖,𝑗,𝑘∣――𝑆𝑖∩――𝑆𝑗∩―――𝑆𝑘∣−⋯+(−1)𝑘−1∣𝑘⋂𝑖=1――𝑆𝑖∣=∑𝑖(𝑘+𝑟−𝑛𝑖−2𝑘−1)−∑𝑖,𝑗(𝑘+𝑟−𝑛𝑖−𝑛𝑗−3𝑘−1)+∑𝑖,𝑗,𝑘(𝑘+𝑟−𝑛𝑖−𝑛𝑗−𝑛𝑘−4𝑘−1)−⋯+(−1)𝑘−1(𝑘+𝑟−∑𝑘𝑖=1𝑛𝑖−𝑘−1𝑘−1)
拿全集 |𝑈| =(𝑘+𝑟−1𝑘−1)
减去上式,得到多重集的组合数
𝐴𝑛𝑠=𝑘∑𝑝=0(−1)𝑝∑𝐴(𝑘+𝑟−1−∑𝐴𝑛𝐴𝑖−𝑝𝑘−1)
其中 A 是充当枚举子集的作用,满足 |𝐴| =𝑝, 𝐴𝑖 <𝐴𝑖+1
。
圆排列
𝑛
个人全部来围成一圈,所有的排列数记为 Q𝑛𝑛
。考虑其中已经排好的一圈,从不同位置断开,又变成不同的队列。 所以有
Q𝑛𝑛×𝑛=A𝑛𝑛⟹Q𝑛=A𝑛𝑛𝑛=(𝑛−1)!
由此可知部分圆排列的公式:
Q𝑟𝑛=A𝑟𝑛𝑟=𝑛!𝑟×(𝑛−𝑟)!
组合数性质 | 二项式推论
由于组合数在 OI 中十分重要,因此在此介绍一些组合数的性质。
(𝑛𝑚)=(𝑛𝑛−𝑚)(1)
相当于将选出的集合对全集取补集,故数值不变。(对称性)
(𝑛𝑘)=𝑛𝑘(𝑛−1𝑘−1)(2)
由定义导出的递推式。
(𝑛𝑚)=(𝑛−1𝑚)+(𝑛−1𝑚−1)(3)
组合数的递推式(杨辉三角的公式表达)。我们可以利用这个式子,在 𝑂(𝑛2)
的复杂度下推导组合数。
(𝑛0)+(𝑛1)+⋯+(𝑛𝑛)=𝑛∑𝑖=0(𝑛𝑖)=2𝑛(4)
这是二项式定理的特殊情况。取 𝑎 =𝑏 =1
就得到上式。
𝑛∑𝑖=0(−1)𝑖(𝑛𝑖)=[𝑛=0](5)![\sum_{i=0}^n(-1)^i\binom{n}{i}=[n=0]\tag{5}]()
二项式定理的另一种特殊情况,可取 𝑎 =1,𝑏 = −1
。式子的特殊情况是取 𝑛 =0
时答案为 1
。
𝑘∑𝑖=0(𝑛𝑖)(𝑚𝑘−𝑖)=(𝑚+𝑛𝑘)(6)
拆组合数的式子,在处理某些数据结构题时会用到。被称为 范德蒙恒等式。
𝑛∑𝑖=0(𝑛𝑖)2=(2𝑛𝑛)(7)
这是 (6)
的特殊情况,取 𝑛 =𝑘 =𝑚
即可。
𝑛∑𝑖=0𝑖(𝑛𝑖)=𝑛2𝑛−1(8)
带权和的一个式子,通过对 (3)
对应的多项式函数求导可以得证。
𝑛∑𝑖=0𝑖2(𝑛𝑖)=𝑛(𝑛+1)2𝑛−2(9)
与上式类似,可以通过对多项式函数求导证明。
𝑛∑𝑙=0(𝑙𝑘)=(𝑛+1𝑘+1)(10)
通过组合分析一一考虑 𝑆 ={𝑎1,𝑎2,⋯,𝑎𝑛+1}
的 𝑘 +1
子集数可以得证,在恒等式证明中比较常用。被称为 朱世杰恒等式。
(𝑛𝑟)(𝑟𝑘)=(𝑛𝑘)(𝑛−𝑘𝑟−𝑘)(11)
通过定义可以证明。
𝑛∑𝑖=0(𝑛−𝑖𝑖)=𝐹𝑛+1(12)
其中 𝐹
是斐波那契数列。
(𝑛+𝑘𝑘)2=𝑘∑𝑗=0(𝑘𝑗)2(𝑛+2𝑘−𝑗2𝑘)(13)
通过 (6)
可以证明。被称为 李善兰恒等式。
二项式反演
记 𝑓𝑛
表示恰好使用 𝑛
个不同元素形成特定结构的方案数,𝑔𝑛
表示从 𝑛
个不同元素中选出 𝑖 ≥0
个元素形成特定结构的总方案数。
若已知 𝑓𝑛
求 𝑔𝑛
,那么显然有:
𝑔𝑛=𝑛∑𝑖=0(𝑛𝑖)𝑓𝑖
若已知 𝑔𝑛
求 𝑓𝑛
,那么:
𝑓𝑛=𝑛∑𝑖=0(𝑛𝑖)(−1)𝑛−𝑖𝑔𝑖
上述已知 𝑔𝑛
求 𝑓𝑛
的过程,就称为 二项式反演。
证明
将反演公式的 𝑔𝑖
展开得到:
𝑓𝑛=𝑛∑𝑖=0(𝑛𝑖)(−1)𝑛−𝑖[𝑖∑𝑗=0(𝑖𝑗)𝑓𝑗]=𝑛∑𝑖=0𝑖∑𝑗=0(𝑛𝑖)(𝑖𝑗)(−1)𝑛−𝑖𝑓𝑗![\begin{aligned}
f_n &= \sum_{i = 0}^{n} \binom{n}{i} (-1)^{n-i} \left[\sum_{j = 0}^{i} \binom{i}{j} f_j\right] \\
&= \sum_{i = 0}^{n}\sum_{j = 0}^{i}\binom{n}{i}\binom{i}{j} (-1)^{n-i}f_j
\end{aligned}]()
先枚举 𝑗
,再枚举 𝑖
,得到:
𝑓𝑛=𝑛∑𝑗=0𝑛∑𝑖=𝑗(𝑛𝑖)(𝑖𝑗)(−1)𝑛−𝑖𝑓𝑗=𝑛∑𝑗=0𝑓𝑗𝑛∑𝑖=𝑗(𝑛𝑖)(𝑖𝑗)(−1)𝑛−𝑖
使用 「组合数性质 | 二项式推论」 的公式 (11) 得到:
𝑓𝑛=𝑛∑𝑗=0𝑓𝑗𝑛∑𝑖=𝑗(𝑛𝑗)(𝑛−𝑗𝑖−𝑗)(−1)𝑛−𝑖=𝑛∑𝑗=0(𝑛𝑗)𝑓𝑗𝑛∑𝑖=𝑗(𝑛−𝑗𝑖−𝑗)(−1)𝑛−𝑖
令 𝑘 =𝑖 −𝑗
。则 𝑖 =𝑘 +𝑗
,上式转换为:
𝑓𝑛=𝑛∑𝑗=0(𝑛𝑗)𝑓𝑗𝑛−𝑗∑𝑘=0(𝑛−𝑗𝑘)(−1)𝑛−𝑗−𝑘1𝑘
使用 「组合数性质 | 二项式推论」 的公式 (5) 得到:
𝑓𝑛=𝑛∑𝑗=0(𝑛𝑗)𝑓𝑗[𝑛=𝑗]=𝑓𝑛![f_n = \sum_{j = 0}^{n}\binom{n}{j}f_j[n = j] = f_n]()
证毕。
本页面最近更新:2025/8/1 13:12:01,更新历史
发现错误?想一起完善? 在 GitHub 上编辑此页!
本页面贡献者:Zhoier, Ir1d, sshwy, Xeonacid, Great-designer, Tiphereth-A, cjsoft, MegaOwIer, Enter-tainer, Marcythm, renbaoshuo, StudyingFather, untitledunrevised, CCXXXI, Chrogeek, Early0v0, Enonya, FFjet, iamtwz, IceySakura, Menci, opsiff, Tiger3018, wq-yang, Xiaobin Ren, XuYueming520, Yukimaikoriya, ZXyaang, zyzzyh
本页面的全部内容在 CC BY-SA 4.0 和 SATA 协议之条款下提供,附加条款亦可能应用