AOJ 2751 Baseball 野球観戦

これはなかなか難しいね

感想

どれも10^6なので、二重ループは回せなさそうです。

したがって、何かしら共通の基準を用意したいのですが。

例えばCのみ(同点の試合)考えるのであればただの組み合わせですよね、N個のものをM個に分ける(重複を考えた)組み合わせはN+M-1個からM-1個選ぶ操作で考えられます。

AとBについて(どちらかのチームが勝つ試合)の処理が面倒なんです。

AとBをどうにかしたいと考えると、両チーム同じ点数を取った後、片方が余分にまた点数を取るという見方ができます。

そこから「全ての試合が同点だったと仮定したときの、各チームがとった点数の合計」を基準としてみると。

全ての試合が同点だと仮定して組み合わせを考えた後、A試合分のチームXが勝つ試合を選び、組み合わせを計算、B試合分のチームYが勝つ試合を選び、組み合わせを計算。という感じで処理できそうな感じがしてきますね。

各事象について、共通部分を抽出するとうまいこと行きますよ、って感じでしょうか。

難しいですね。

コード

#include <bits/stdc++.h>
 
using namespace std;
 
#ifdef DEBUG_MODE
    #define DBG(n) n;
#else
    #define DBG(n) ;
#endif
#define REP(i,n) for(ll (i) = (0);(i) < (n);++i)
#define rep(i,s,g) for(ll (i) = (s);(i) < (g);++i)
#define rrep(i,s,g) for(ll (i) = (s);i >= (g);--(i))
#define PB push_back
#define MP make_pair
#define FI first
#define SE second
#define SHOW1d(v,n) {for(int WWW = 0;WWW < (n);WWW++)cerr << v[WWW] << ' ';cerr << endl << endl;}
#define SHOW2d(v,i,j) {for(int aaa = 0;aaa < i;aaa++){for(int bbb = 0;bbb < j;bbb++)cerr << v[aaa][bbb] << ' ';cerr << endl;}cerr << endl;}
#define ALL(v) v.begin(),v.end()
#define Decimal fixed<<setprecision(20)
#define INF 1000000000
#define LLINF 1000000000000000000LL
#define MOD 1000000007
 
typedef long long ll;
typedef pair<ll,ll> P;
 
//aをbで割る
long long mod_div(ll a,ll b){
    if(a % b == 0)return a/b;
    long long tmp = MOD - 2,c = b,ret = 1;
    while(tmp > 0){
        if(tmp & 1){
            ret *= c;ret %= MOD;
        }
        c *= c;c %= MOD;tmp >>= 1;
    }
    return a*ret%MOD;
}
  
#define MAX_K 3333333
vector<long long> kaijo(MAX_K);
long long combination(long long n, long long r){
    if(n < r || n < 0 || r < 0)
        return 0;
    if(kaijo[0] != 1){
        kaijo[0] = 1;
        for(long long i = 1;i < MAX_K;i++)kaijo[i] = (kaijo[i-1] * i) % MOD;
    }
    long long ret = kaijo[n];
    long long tmp = (kaijo[r] * kaijo[n-r]) % MOD;
    return mod_div(ret,tmp);
}
 
int main(){
    ll a,b,c,x,y;
     
    while(cin >> a >> b >> c >> x >> y,a+b+c+x+y){
         
        ll ans = 0;
        ll seica = combination(a+b+c,c) * combination(a+b,b) % MOD;
        for(ll i = 0;i <= min(x,y);i++){
            ll X = x-i-a;
            ll Y = y-i-b;
            if(X < 0 || y < 0)continue;
            if(a == 0 && X > 0)continue;
            if(b == 0 && Y > 0)continue;
            ll tmp = seica;
            if(a+b+c+i-1 >= i)tmp = (tmp * combination(a+b+c+i-1,i)) % MOD;
            if(a+X-1 >= X)tmp = (tmp * combination(a+X-1,X)) % MOD;
            if(b+Y-1 >= Y)tmp = (tmp * combination(b+Y-1,Y)) % MOD;
            ans = (ans+tmp) % MOD;
        }
         
        cout << ans << endl;
    }
    return 0;
}

コード自体は短いですね。