bzoj千题计划261:bzoj3294: [Cqoi2011]放棋子

http://www.lydsy.com/JudgeOnline/problem.php?id=3294

如果一个颜色的棋子放在了第i行第j列,那这种颜色就会占据第i行第j列,其他颜色不能往这儿放

设第k种颜色的棋子有a[k]个

令g[k][i][j] 表示第k种颜色的棋子,恰好占据i行j列的方案数

g[k][i][j]=C(i*j,a[k])-Σh Σl g[h][l]*C(i,h)*C(j,l)  1<=h<=i,1<=l<=j,且满足 h!=i 或 l !=j

即 总方案数(在i*j个格子中选a[k]个) 减去 没有恰好占据i行j列的方案数

令f[k][i][j] 表示前k种颜色的棋子,放完之后,还剩下i行j列的方案数

f[k][i][j]= Σh Σl f[k-1][h][l]*g[k][h-i][l-j]*C[h][h-i]*C[l][l-j]  i<h<=n,j<l<=m

即 枚举前k-1种颜色的棋子放完后,剩下h行l列,那么 第k种颜色就占据h-i行l-j列

#include<cstdio>

using namespace std;

const int mod=1e9+;

#define min(x,y) x<y ? x : y

#define N 31
#define M 11 int C[N*N][N*N]; int a[M];
long long g[M][N][N],f[M][N][N]; int main()
{
int n,m,c;
scanf("%d%d%d",&n,&m,&c);
for(int i=;i<=c;++i) scanf("%d",&a[i]);
int lim=n*m;
C[][]=;
for(int i=;i<=lim;++i)
{
C[i][]=;
for(int j=;j<=i;++j) C[i][j]=(C[i-][j-]+C[i-][j])%mod;
}
int r1,r2;
for(int k=;k<=c;++k)
{
r1=min(a[k],n);
for(int i=;i<=r1;++i)
{
r2=min(a[k],m);
for(int j=;j<=r2;++j)
if(i*j>=a[k])
{
g[k][i][j]=C[i*j][a[k]];
for(int h=;h<=i;++h)
for(int l=;l<=j;++l)
if((h!=i || l!=j) && h*l>=a[k])
{
g[k][i][j]-=g[k][h][l]*C[i][h]%mod*C[j][l]%mod;
if(g[k][i][j]<) g[k][i][j]+=mod;
}
}
}
}
f[][n][m]=;
for(int k=;k<=c;++k)
for(int i=;i<n;++i)
for(int j=;j<m;++j)
for(int h=i+;h<=n;++h)
for(int l=j+;l<=m;++l)
if((h-i)*(l-j)>=a[k])
f[k][i][j]=(f[k][i][j]+f[k-][h][l]*g[k][h-i][l-j]%mod*C[h][h-i]%mod*C[l][l-j]%mod)%mod;
int ans=;
for(int i=;i<n;++i)
for(int j=;j<m;++j)
{
ans+=f[c][i][j];
if(ans>=mod) ans-=mod;
}
printf("%d",ans);
return ;
}

3294: [Cqoi2011]放棋子

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 797  Solved: 319
[Submit][Status][Discuss]

Description

 

Input

输入第一行为两个整数n, m, c,即行数、列数和棋子的颜色数。
第二行包含c个正整数,即每个颜色的棋子数。
所有颜色的棋子总数保证不超过nm。
N,M<=30 C<=10 总棋子数有大于250的情况
 

Output

输出仅一行,即方案总数除以 1,000,000,009的余数。

Sample Input

4 2 2
3 1

Sample Output

8
上一篇:移动端click事件延时


下一篇:PHP学习笔记十四【面向对象】