CodeForces 57C Array 组合计数+逆元

题目链接:

http://codeforces.com/problemset/problem/57/C

题意:

给你一个数n,表示有n个数的序列,每个数范围为[1,n],叫你求所有非降和非升序列的个数。

题解:

由于对称性,我们只要求非降序的个数就可以了(n个数全部相等的情况既属于非升也属于非降)

我们在满足条件的n个数之前加一个虚节点1,在第n个数之后加一个虚节点n,那么考虑这n+2个数组成的非降序列:

假设序列里的第i个数为a[i],我们设xi=a[i+1]-a[i]+1,1<=i<=n+1,则满足每个数>=1,且sum(x[1],x[2]...x[n+1])=2*n;

那么相当于求将2*n分成n个部分,且每个部分的值大于等于1,则易得非降序列总数为:C(n,2*n-1)(2*n-1 选 n)

所以最后的答案是2*C(n,2*n-1)-n;

代码:

 1 #include<iostream>
 2 #include<cstring>
 3 #include<cstdio>
 4 using namespace std;
 5 
 6 const int mod = 1000000007;
 7 typedef long long LL;
 8 int n;
 9 //扩展欧几里得
10 void gcd(int a, int b, int &d, int &x, int &y) {
11     if (!b) { d = a; x = 1; y = 0; }
12     else { gcd(b, a%b, d, y, x); y -= x*(a / b); }
13 }
14 //求逆元
15 int inv(int a) {
16     int d, x, y;
17     gcd(a, mod, d, x, y);
18     return x;
19 }
20 //求阶乘
21 int solve(int _n,int x) {
22     LL ret = 1;
23     while (_n--) {
24         ret *= x;
25         ret %= mod;
26         x--;
27     }
28     return ret;
29 }
30 
31 int main() {
32     while (scanf("%d", &n) == 1 && n) {
33         int ans = (LL)solve(n, 2 * n - 1)*inv(solve(n,n))%mod;
34         ans = ans * 2 - n;
35         ans = (ans%mod + mod) % mod;
36         printf("%d\n", ans);
37     }
38     return 0;
39 }

 

posted @ 2016-05-05 01:18  fenicnn  阅读(357)  评论(0编辑  收藏  举报