[ZPG TEST 111] 奶牛的新家【DP】
3.奶牛的新家
【问题描述】
由于奶牛们纷纷表示破旧的房子实在是太丑陋了,DD决定给他们建造新家。现在有许多奶牛决定将家建造在n*m的城市中。然而奶牛们分成了k帮派,不同帮派的奶牛不能住在同列或同行上。现在DD想知道一共有多少建造方案。
【输入】
第一行三个整数n,m,k
接下来一行k个整数,分别表示每个帮派有多少只牛
【输出】
一行一个整数,建造方案数mod 100000007
【样例输入】
2 3 2
1 1
【样例输出】
12
【输入输出样例说明】
两头不同帮派的牛放在2*3的图中的方案数为12
【数据范围】
对于30%的数据,n,m≤4
对于80%的数据,n,m≤20
对于100%的数据,n,m≤30;k≤10
这是我们第二次六校联测(标题是“多校联测3”),又挂了,唉,又是死在了dp上了,烦透了。
先观察这一题,一种颜色一定是占据了若干行和若干列的,在这些行与列上是不能放别的颜色的。令g(i, j, k)表示第k种颜色恰好占据了i行j列的方案数,这里“恰好”是指这i行j列中没有哪一行或者哪一列是空的。那么有:
g(i, j, k) = C(i * j, a[k]) - sigma ( g(i', j', k) * C(i, i') * C(j, j') ),其中要满足i' < i或 j' < j,i * j要 >= a[k](a[k]表示第k种颜色的个数),C表示的是组合。
上面那条方程的意思就是,首先要加上C(i * j, a[k])表示在这i * j个格子里放a[k]个,那么有C(i * j, a[k])种方案。减掉的那一部分的意义是:某些方案并不满足这种状态(注意状态表示的是“恰好占据”),所以需要减掉这些状态。
然后f(i, j, k)表示前k种颜色,恰好占据了i行j列的方案数,那么有:
f(i, j, k) = sigma( f(i - i', j - j', k - 1) * g(i', j') * C(i, i') * C(j, j') )
#include <cstdio> const long long mod = 100000007; int n, m, K, a[15];
long long c[1005][1005], f[35][35][15], g[35][35][15], ans; int main(void) {
scanf("%d%d%d", &n, &m, &K);
for (int i = 1; i <= K; ++i) {
scanf("%d", a + i);
}
c[0][0] = 1;
for (int i = 1; i < 1001; ++i) {
c[i][0] = 1;
for (int j = 1; j <= i; ++j) {
c[i][j] = (c[i - 1][j - 1] + c[i - 1][j]) % mod;
}
} for (int k = 1; k <= K; ++k) {
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
if (i * j < a[k]) {
continue;
}
g[i][j][k] = c[i * j][a[k]];
for (int i_ = 1; i_ <= i; ++i_) {
for (int j_ = 1; j_ <= j; ++j_) {
if (i_ < i || j_ < j) {
g[i][j][k] = (g[i][j][k] - (g[i_][j_][k] * c[i][i_] % mod * c[j][j_] % mod) + mod) % mod;
}
}
}
}
}
} f[0][0][0] = 1;
for (int k = 1; k <= K; ++k) {
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
if (i * j < a[k]) {
continue;
}
for (int i_ = 1; i_ <= i; ++i_) {
for (int j_ = 1; j_ <= j; ++j_) {
f[i][j][k] = (f[i][j][k] + f[i - i_][j - j_][k - 1] * g[i_][j_][k] % mod * c[i][i_] % mod * c[j][j_]) % mod;
}
}
}
}
} for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
ans = (ans + f[i][j][K] * c[n][i] % mod * c[m][j] % mod) % mod;
}
}
printf("%d\n", (int)ans);
return 0;
}
[ZPG TEST 111] 奶牛的新家【DP】的更多相关文章
- 【BZOJ-3631】松鼠的新家       树形DP?+ 倍增LCA + 打标记
		3631: [JLOI2014]松鼠的新家 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1231 Solved: 620[Submit][Stat ... 
- 3631. [JLOI2014]松鼠的新家【树形DP】
		Description 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在“树”上.松鼠想邀 ... 
- 刷题总结——松鼠的新家(bzoj3631)
		题目: Description 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在“树”上. ... 
- 此博客主人已搬家访问新家地址:http://write.blog.csdn.net/postlist
		此博客主人已搬家访问新家地址:http://write.blog.csdn.net/postlist 
- BZOJ 3631 【JLOI2014】 松鼠的新家
		Description 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在"树&q ... 
- 【bzoj3631】[JLOI2014]松鼠的新家
		题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他居然真的住在"树"上.松 ... 
- 【BZOJ3631】松树的新家 树链剖分
		BZOJ3631 松树的新家 Description 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他 ... 
- BZOJ 3631: [JLOI2014]松鼠的新家( 树链剖分 )
		裸树链剖分... ------------------------------------------------------------------- #include<bits/stdc++ ... 
- 3631: [JLOI2014]松鼠的新家
		3631: [JLOI2014]松鼠的新家 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 707 Solved: 342[Submit][Statu ... 
随机推荐
- TeamCity - Docker创建
			// 创建Server docker run -it --name teamcity-server-instance \-v /home/tc_datadir:/data/teamcity_serve ... 
- oracle rac 安装错误整理。
			今天是2014.05.26,离别N久的博客今天继续使用. 近期一直忙着离职.入职另外加上家的网一直没有交费,弄的自己開始不那么安稳.学习就是须要一种心情平静.内心稳妥的去进行. 因换笔记本,特须要又一 ... 
- 解决安装oracle11g r2时提示pdksh conflicts with ksh-20100621-2.el6.i686问题
			http://blog.csdn.net/linghao00/article/details/7943740 http://www.2cto.com/os/201306/218566.html 在Ce ... 
- Deepin-安装和卸载软件
			一般默认厂商源安装软件 安装软件: 示例:sudo apt-get install xx 实例:sudo apt-get install nodejs 卸载软件: 示例:sudo apt-get -- ... 
- activiti自己定义流程之自己定义表单(二):创建表单
			注:环境配置:activiti自己定义流程之自己定义表单(一):环境配置 在上一节自己定义表单环境搭建好以后,我就正式開始尝试自己创建表单,在后台的处理就比較常规,主要是针对ueditor插件的功能在 ... 
- robotframework接口自动化
			robot framework框架在测试接口上比soapUI好用的多,在此介绍下get方法的HTTP接口,其实这个接口也是把POST数据作为参数进行get请求,使用post 方法也是一样,一共6步就可 ... 
- 跟面试官讲Binder(零)
			面试的时候,面试官问你说,简单说一下Android的Binder机制,你会怎么回答? 我想,我会这么说. 在Android启动的时候,Zygote进程孵化出第一个子进程叫SystemServer,而在 ... 
- windows下Python扩展问题error: Unable to find vcvarsall.bat
			由于对于Windows下Python扩展不熟,今天遇到一个安装问题,特此做个tag.解决方式在stackoverflow上,网址例如以下: http://stackoverflow.com/quest ... 
- mybatis_2
			<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE configuration PUBLIC ... 
- YTU 2928: 取不重复的子串。
			2928: 取不重复的子串. 时间限制: 1 Sec 内存限制: 128 MB 提交: 5 解决: 5 题目描述 输入字母构成的字符串(不大于30字符)从中读取3个不重复的字符,求所有取法,取出的 ... 
