求组合数Ⅱ
1万组数据,
1
≤
b
≤
a
≤
1
0
5
1 \le b \le a \le 10^5
1≤b≤a≤105,预处理阶乘。时间复杂度
O
(
N
l
o
g
N
)
O(NlogN)
O(NlogN)
C
a
b
=
a
!
(
b
−
a
)
!
b
!
C_a^b = \frac{a !}{(b - a)! b!}
Cab=(b−a)!b!a!
预处理出
i
!
i !
i!和
(
i
!
)
−
1
(i !)^{-1}
(i!)−1
f
a
c
t
[
i
]
=
i
!
m
o
d
(
1
0
9
+
7
)
i
n
f
a
c
t
[
i
]
=
(
i
!
)
−
1
fact[i] = i ! mod (10^9 + 7) \\ infact[i] = (i!)^{-1}
fact[i]=i!mod(109+7)infact[i]=(i!)−1
因此
C
a
b
=
f
a
c
t
[
a
]
×
i
n
f
a
c
t
[
b
−
a
]
×
i
n
f
a
c
t
[
b
]
C_a^b = fact[a] \times infact[b - a] \times infact[b]
Cab=fact[a]×infact[b−a]×infact[b]
题目
题目描述:
给定n组询问,每组询问给定两个整数a,b,请你输出C(a,b) mod (10^9+7)的值。
输入格式
第一行包含整数n。接下来n行,每行包含一组a和b。
输出格式
共n行,每行输出一个询问的解。
数据范围
1≤n≤100000,1≤b≤a≤10^5
输入样例:
3
3 1
5 3
2 2
输出样例:
3
10
1
代码
#include<iostream>
using namespace std;
typedef long long LL;
const int N = 100010, mod = 1e9 + 7;
int fact[N], infact[N];
//快速幂
int qmi(int a, int k, int p) {
int res = 1;
while(k) {
if(k & 1) res = (LL) res * a % p;
a = (LL) a * a % p;
k >>= 1;
}
return res;
}
int main() {
fact[0] = infact[0] = 1;
for(int i = 1; i < N; i ++ ) {
fact[i] = (LL)fact[i - 1] * i % mod;
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;
}