码迷,mamicode.com
首页 > 系统相关 > 详细

lightoj 1226 - One Unit Machine(dp+大组合数去摸)

时间:2017-10-12 17:09:45      阅读:272      评论:0      收藏:0      [点我收藏+]

标签:任务   machine   main   位置   scanf   int   ase   namespace   col   

题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1226

 

题解:由于这些任务完成是有先后的所以最后一个完成的肯定是最后一个任务的子任务,不妨设dp[i]表示第几个任务完成后总共有几种方案,这里要逆着来至于为什么想想也是挺好理解的。于是有这么一个方程式dp[i]=dp[i + 1]*C(sum-1,k[i]-1),这样列出来就更好理解了。就是最后一个位置肯定是确定的之后就靠组合来凑。

#include <iostream>
#include <cstring>
#include <cstdio>
#define mod 1000000007
using namespace std;
typedef long long ll;
const int M = 1234;
ll k[M];
ll dp[M] , Po[1234567];
ll mod_pow(ll a , ll b) {
    ll res = 1;
    while(b) {
        if(b & 1) res = (res * a) % mod;
        a = (a * a) % mod;
        b >>= 1;
    }
    return res;
}
ll C(ll n , ll m) {
    if(n < m) return 0;
    if(!m || !n) return 1;
    ll res = (Po[n] * mod_pow(Po[m] * Po[n - m] % mod , mod - 2)) % mod;
    return res;
}
int main() {
    int t , n;
    scanf("%d" , &t);
    int Case = 0;
    Po[0] = 1;
    for(int i = 1 ; i < 1234567 ; i++) {
        Po[i] = Po[i - 1] * i % mod;
        Po[i] %= mod;
    }
    while(t--) {
        memset(dp , 0 , sizeof(dp));
        scanf("%d" , &n);
        ll sum = 0;
        for(int i = 1 ; i <= n ; i++) {
            scanf("%lld" , &k[i]);
            sum += k[i];
        }
        dp[n] = C(sum - 1 , k[n] - 1);
        dp[n] %= mod;
        sum -= k[n];
        for(int i = n - 1 ; i >= 1 ; i--) {
            dp[i] = dp[i + 1] * C(sum - 1 , k[i] - 1) % mod;
            dp[i] %= mod;
            sum -= k[i];
        }
        printf("Case %d: %lld\n" , ++Case , dp[1]);
    }
    return 0;
}

 

lightoj 1226 - One Unit Machine(dp+大组合数去摸)

标签:任务   machine   main   位置   scanf   int   ase   namespace   col   

原文地址:http://www.cnblogs.com/TnT2333333/p/7656581.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!