组合数
这几天的竞赛期末考试中遇到了组合数的题目,就是 NOIP2016 组合数问题 原题,正好现在数学学到了一点点排列组合,就来整理一下吧。
组合数
就是从 \(n\) 个不同元素中,抽取 \(m\) 个元素的方案数。
组合数公式为:
\[\dbinom{n}{m}= \dfrac{n!}{m! \times (n-m)!}\]
通过杨辉三角可得到递推公式:
\(\dbinom{i}{j} = \dbinom{i-1}{j-1} + \dbinom{i-1}{j}\)
还有一个递推公式(可证明):
\[\dbinom{i}{j} = \dfrac{n-k+1}{k} \times \dbinom{i}{j-1}\]
但是这个可能会乘法溢出
证明
证明完毕
题目思路
暴力
暴力,暴力,暴力!
预处理
用第一个递推公式先递推出来所有的数,然后对于每一次查询进行判断是否 % k == 0,时间复杂度为 \(\mathcal{O}(q+tn^2)\),还是有点慢
预处理优化
我们可以想到每一判断都 % k 有点慢所以可以在预处理时 % k,判断是否等于 0 ,开过 \(O2\) 95 pts,还需要优化。
前缀和+预处理
我们可以二维前缀和等于 0 的个数,然后每一次查询即可,时间复杂度变成了 \(\mathcal{O}(q+t)\) ,可以过了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43#include<cstdio>
#include<iostream>
using namespace std;
const int p = 1000;
int c[2010][2010] = {0};
int f[2010][2010] = {0};
int n,m,k,t;
void complet(){
c[0][0] = 1;
c[1][0] = 1;
c[1][1] = 1;
for(int i=0;i<=2000;i++){
c[i][0] = 1;
c[i][i] = 1;
for(int j=1;j<i;j++){
c[i][j] = (c[i-1][j-1] + c[i-1][j]) % k;
}
}
for (int i=1;i<2000;i++){
for (int j=1;j<=i;j++){
f[i][j] = f[i-1][j] + f[i][j-1] - f[i-1][j-1]; \\ 前缀和
if(c[i][j] == 0) f[i][j] ++;
}
f[i][i+1] = f[i][i];
}
}
int main()
{
int cnt = 0;
cin>>t>>k;
complet();
while(t--){
cnt = 0;
scanf("%d %d",&n,&m);
if(m > n)
m = n;
printf("%d\n",f[n][m]);
}
return 0;
}
本博客所有文章除特别声明外,均采用 CC BY-SA 4.0 协议 ,转载请注明出处!