题目描述:

输入格式

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

输出格式

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

样例

样例输入

4 2 2
3 1

样例输出

8

数据范围与提示

30% n,m<=10

solution:10%:cout<<0<<endl;

肯定有0的情况比如c>min(n,m)之类的。。。

20%:搜索,枚举所有状态。

据说这是搜索最高得分,然而博主考试时只拿到10分,而且还不是TLE。


下面说正解

我们考虑dp,设f[i][j][k]表示前k种颜色的棋子占领任意i行j列的方案数,g[i][j][k]表示第k种颜色的所有棋子占领任意i行j列的方案数;

那么我们首先可以得到g[i][j][k]=$C_{i*j}^{num[k]}$-$\sum_\limits{p=1}^{i}$$\sum_\limits{q=1}^{j}$g[p][q][k]*$C_{i}^{p}$*$C_{j}^{q}$

其实就是用合法的减去不合法的(实际上有没有被占领的行或列的方案数)

接下来得到f的方程:

$f[i][j][k]=\sum_{p=0}^{i-1}\sum_{q=0}^{j-1}$

$f[i][j][k]=\sum_{p=0}^{i-1}\sum_{q=0}^{j-1}f[p][q][k-1]*g[i-p][j-q][k]*C_{n-p}^{i-p}*C_{m-q}^{j-q}$,f[0][0][0]=1;

p,q,k-1就是枚举的上一个状态,$C_{n-p}^{i-p}$表示n-p行中选出i-p行放棋子,$C_{m-q}^{i-q}$同理,

最后ans=$\sum_{i=1}^{n}$$\sum_{j=1}^{m}$f[i][j][c],于是这道题就完美地解决了

放代码:

 #include<iostream>
#include<cstdio>
#include<cstring>
#define mod 1000000009
#define ll long long
#define MAXNM 905
using namespace std;
int n,m,c,num[];
ll C[MAXNM][MAXNM],g[][][],f[][][],ans=;
int main(){
scanf("%d%d%d",&n,&m,&c);
for(int i=;i<=n*m;i++){
C[i][]=;
for(int j=;j<=i;j++)
C[i][j]=(C[i-][j]+C[i-][j-])%mod;
}
f[][][]=;
for(int k=;k<=c;k++)
scanf("%d",&num[k]);
for(int k=;k<=c;k++){
for(int i=;i<=n;i++)
for(int j=;j<=m;j++){
if(i*j<num[k]) continue;
g[i][j][k]=C[i*j][num[k]];
for(int p=;p<=i;p++)
for(int q=;q<=j;q++){
if(p<i||q<j)
g[i][j][k]=(g[i][j][k]-g[p][q][k]*C[i][p]%mod*C[j][q]%mod)%mod;
//cout<<g[i][j]<<endl;
}
}
}
for(int k=;k<=c;k++){
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
for(int p=;p<i;p++)
for(int q=;q<j;q++){
int l=i-p,r=j-q;
if(l*r<num[k]) continue;
f[i][j][k]=(f[i][j][k]+f[p][q][k-]*g[l][r][k]%mod*C[n-p][l]%mod*C[m-q][r]%mod)%mod;
//cout<<f[i][j][k]<<endl;
}
}
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
ans=(ans+f[i][j][c])%mod;
printf("%lld\n",ans);
return ;
}

我们发现g只对当前一种棋子有贡献,所以第三维可以干掉,在每次输入时处理g和f

#include<iostream>
#include<cstdio>
#include<cstring>
#define mod 1000000009
#define ll long long
#define MAXNM 905
using namespace std;
int n,m,c,num[12];
ll C[MAXNM][MAXNM],g[35][35],f[35][35][15],ans=0;
int main(){
scanf("%d%d%d",&n,&m,&c);
for(int i=0;i<=n*m;i++){
C[i][0]=1;
for(int j=1;j<=i;j++)
C[i][j]=(C[i-1][j]+C[i-1][j-1])%mod;
}
f[0][0][0]=1;
for(int k=1;k<=c;k++){
scanf("%d",&num[k]);
memset(g,0,sizeof(g));
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++){
if(i*j<num[k]) continue;
g[i][j]=C[i*j][num[k]];
for(int p=1;p<=i;p++)
for(int q=1;q<=j;q++){
if(p<i||q<j)
g[i][j]=(g[i][j]-g[p][q]*C[i][p]%mod*C[j][q]%mod)%mod;
//cout<<g[i][j]<<endl;
}
}
for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
for(int p=0;p<i;p++)
for(int q=0;q<j;q++){
int l=i-p,r=j-q;
if(l*r<num[k]) continue;
f[i][j][k]=(f[i][j][k]+f[p][q][k-1]*g[l][r]%mod*C[n-p][l]%mod*C[m-q][r]%mod)%mod;
//cout<<f[i][j][k]<<endl;
}
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
ans=(ans+f[i][j][c])%mod;
printf("%lld\n",ans);
return 0;
}

[CQOI2011]放棋子--DP的更多相关文章

  1. bzoj3294[Cqoi2011]放棋子 dp+组合+容斥

    3294: [Cqoi2011]放棋子 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 755  Solved: 294[Submit][Status] ...

  2. [CQOI2011]放棋子 (DP,数论)

    [CQOI2011]放棋子 \(solution:\) 看到这道题我们首先就应该想到有可能是DP和数论,因为题目已经很有特性了(首先题面是放棋子)(然后这一题方案数很多要取模)(而且这一题的数据范围很 ...

  3. 【BZOJ 3294】 3294: [Cqoi2011]放棋子 (DP+组合数学+容斥原理)

    3294: [Cqoi2011]放棋子 Description Input 输入第一行为两个整数n, m, c,即行数.列数和棋子的颜色数.第二行包含c个正整数,即每个颜色的棋子数.所有颜色的棋子总数 ...

  4. P3158 [CQOI2011]放棋子(dp+组合数)

    P3158 [CQOI2011]放棋子 放棋子的顺序和方案数无关,所以可以从按颜色递推 设$f[u][p][k]$为放到第$u$种颜色,所剩空间$p*k$的方案数 $g[u][i][j]$表示第$u$ ...

  5. [洛谷P3158] [CQOI2011]放棋子

    洛谷题目链接:[CQOI2011]放棋子 题目描述 在一个m行n列的棋盘里放一些彩色的棋子,使得每个格子最多放一个棋子,且不同 颜色的棋子不能在同一行或者同一列.有多少祌方法?例如,n=m=3,有两个 ...

  6. BZOJ 3294: [Cqoi2011]放棋子

    3294: [Cqoi2011]放棋子 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 628  Solved: 238[Submit][Status] ...

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

    http://www.lydsy.com/JudgeOnline/problem.php?id=3294 如果一个颜色的棋子放在了第i行第j列,那这种颜色就会占据第i行第j列,其他颜色不能往这儿放 设 ...

  8. BZOJ3294: [Cqoi2011]放棋子

    Description   Input 输入第一行为两个整数n, m, c,即行数.列数和棋子的颜色数.第二行包含c个正整数,即每个颜色的棋子数.所有颜色的棋子总数保证不超过nm. Output 输出 ...

  9. BZOJ3294: [Cqoi2011]放棋子(计数Dp,组合数学)

    题目链接 解题思路: 发现一个性质,如果考虑一个合法的方案可以将行和列都压到一起,也就是说,在占用行数和列数一定的情况下,行列互换是不会影响答案的,那么考虑使用如下方程: $f[i][j][k]$为占 ...

随机推荐

  1. js对象 事件

     对象  创建  var   myObject = {};/* 声明对象字面变量*/ 添加值myObject.name="Jener";myObject.age=25; 代码格式 ...

  2. jquery选择器中中>和空格的区别

    空格:$('parent childchild')表示获取parent下的所有的childchild节点 大于号:$('parent > childchild')表示获取parent下的所有下一 ...

  3. PHP curl采集

    if (function_exists('curl_init')) { //检查函数是否存在 $url = "http://***.com/"; $ch = curl_init() ...

  4. spss logistic回归分析结果如何分析

    spss logistic回归分析结果如何分析 如何用spss17.0进行二元和多元logistic回归分析 一.二元logistic回归分析 二元logistic回归分析的前提为因变量是可以转化为0 ...

  5. day 70 Django基础五之django模型层(二)多表操作

    Django基础五之django模型层(二)多表操作   本节目录 一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询.分组查询.F查询和Q查询 六 ORM ...

  6. Han Xin and His Troops(扩展中国剩余定理 Python版)

    Han Xin and His Troops(扩展中国剩余定理 Python版) 题目来源:2019牛客暑期多校训练营(第十场) D - Han Xin and His Troops 题意:   看标 ...

  7. USACO training course Mother's Milk /// DFS(有点意思) oj10120

    题目大意: 输入 A B C 为三个容器的容量 一开始A B是空的 C是满的 每一次倾倒只能在 盛的容器满 或 倒的容器空 时才停止 输出当A容器空时 C容器内剩余量的所有可能值 Sample Inp ...

  8. linux 文件类型的颜色

    linux文件颜色的含义:蓝色代表目录 绿色代表可执行文件 红色表示压缩文件 浅蓝色表示链接文件 灰色表示其他文件 红色闪烁表示链接的文件有问题了 黄色表示设备文件 蓝色文件----------目录 ...

  9. 警告(alert 消息对话框) 如果你不点击“确定”,就不能对网页做任何操作,这个小窗口就是使用alert实现的

    警告(alert 消息对话框) 我们在访问网站的时候,有时会突然弹出一个小窗口,上面写着一段提示信息文字.如果你不点击"确定",就不能对网页做任何操作,这个小窗口就是使用alert ...

  10. C++: inheritance

    公有继承(public).私有继承(private).保护继承(protected)是常用的三种继承方式. 1. 公有继承(public) 公有继承的特点是基类的公有成员和保护成员作为派生类的成员时, ...