• codeforces #309 div1 A


    先说我的解法吧

    首先设f(i,j)表示选了前i个球且j种颜色都已经选完了的方案数

    这显然是可以随便转移的

    #include<cstdio>
    #include<cstring>
    #include<cstdlib>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    
    typedef long long LL;
    const int mod=1000000007;
    const int maxn=1010;
    int n,m;
    int Num[maxn],sum[maxn];
    int f[maxn][maxn];
    int C[maxn][maxn];
    int S[maxn][maxn];
    //f(i,j)��ʾѡ����i����ǰj����ɫ��ѡ��ķ����� 
    
    void pre_C(){
    	C[0][0]=1;
    	for(int i=1;i<=1000;++i){
    		C[i][0]=C[i][i]=1;
    		for(int j=1;j<i;++j){
    			C[i][j]=C[i-1][j-1]+C[i-1][j];
    			if(C[i][j]>=mod)C[i][j]-=mod;
    		}
    	}return;
    }
    
    int main(){
    	scanf("%d",&n);
    	for(int i=1;i<=n;++i)scanf("%d",&Num[i]),m+=Num[i],sum[i]=sum[i-1]+Num[i];
    	pre_C();
    	f[0][0]=1;S[0][0]=1;
    	for(int i=1;i<=m;++i)S[i][0]=1;
    	for(int j=1;j<=n;++j){
    		for(int i=1;i<=m;++i){
    			if(i>=sum[j])f[i][j]=1LL*S[i-1][j-1]*C[i-1-sum[j-1]][Num[j]-1]%mod;
    			else f[i][j]=0;
    			S[i][j]=f[i][j]+S[i-1][j];
    			if(S[i][j]>=mod)S[i][j]-=mod;
    		}
    	}cout<<f[m][n]<<endl;
    	return 0;
    	
    }
    

      

    还有一种做法是考虑最后一种颜色,一定有一个在最后一位,其余的任意放

    之后考虑倒数第二种颜色,一定有一个在当前的最后一位,其余的任意放

    以此类推

    所以ans=C(sum-1,c[n]-1]*C(sum-c[n]-1,c[n-1]-1)......

  • 相关阅读:
    Redis进阶
    redis常用指令
    MarkDown基本语法
    JAVA多线程面试
    使用POI操作Excel
    IDEA+GIT的使用
    获取地址栏的参数
    mybatis逆向工程
    遍历map集合
    springboot批量删除
  • 原文地址:https://www.cnblogs.com/joyouth/p/5361977.html
Copyright © 2020-2023  润新知