洛谷题目链接:[CQOI2011]放棋子

题目描述

在一个m行n列的棋盘里放一些彩色的棋子,使得每个格子最多放一个棋子,且不同

颜色的棋子不能在同一行或者同一列。有多少祌方法?例如,n=m=3,有两个白棋子和一

个灰棋子,下面左边两祌方法都是合法的,但右边两祌都是非法的。

输入输出格式

输入格式:

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

输出格式:

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

输入输出样例

输入样例#1:

4 2 2

3 1

输出样例#1:

8

说明

N,M<=30 C<=10 总棋子数<=250

题解: 一道\(DP\)的好题.

定义状态\(f[i][j][k]\)表示用前\(k\)种颜色占领了任意\(i\)行\(j\)列.

设\(a[k]\)表示第\(k\)种颜色的棋子的个数.

转移很显然是$$f[i][j][k]=\sum_{l=0}{i-1}\sum_{r=0}{j-1}f[l][r][k-1]C_{n-l}{i-l}*C_{m-r}{j-r}用a[k]个棋子占领任意i-l行j-r列的方案数$$

那么我们再定义状态\(g[i][j][k]\)表示用\(k\)个相同颜色的棋子占领任意\(i\)行\(j\)列的方案数,直接算不太好算,我们可以考虑容斥:$$g[i][j][k]=C_{ij}k-\sum_{l=0}{i}\sum_{r=0}{j}g[l][r][k]*C_{i}{l}C_{j}^{r},(l \not= i \ ||\ r \not= j)$$

预处理了\(g\)数组,就可以对\(f\)数组转移了:$$f[i][j][k]=\sum_{l=0}{i-1}\sum_{r=0}{j-1}f[l][r][k-1]C_{n-l}{i-l}*C_{m-r}{j-r}g[i-l][j-r][a[k]]$$

因为不一定要放满整个棋盘,所以$$ans=\sum_{l=1}n\sum_{r=1}mf[i][j][c]$$

其实还可以用滚动数组滚掉\(g\)数组的最后一维.

// luogu-judger-enable-o2
#include<bits/stdc++.h>
using namespace std;
const int N = 30+5;
const int COL = 10+5;
const int K = 250+5;
const int mod = 1e9+9; int n, m, C, a[N], c[1000][1000], ans = 0;
int f[N][N][COL], g[N][N][1000];
// f : k types of col occupied any i lines, j rows
// g : same type k chess piece occupied any i lines, j rows int main(){
cin >> n >> m >> C;
for(int i = 1; i <= n; i++) cin >> a[i];
f[0][0][0] = c[0][0] = 1;
for(int i = 1; i <= 900; 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;
}
for(int k = 1; k <= C; k++)
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++){
if(a[k] > i*j) continue; int res = 0;
g[i][j][a[k]] = c[i*j][a[k]];
for(int l = 1; l <= i; l++)
for(int r = 1; r <= j; r++)
if(l < i || r < j) (res += 1ll*c[i][l]*c[j][r]%mod*g[l][r][a[k]]%mod) %= mod;
g[i][j][a[k]] = (g[i][j][a[k]]-res+mod)%mod;
}
for(int k = 1; k <= C; k++)
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++)
for(int l = 0; l < i; l++)
for(int r = 0; r < j; r++)
(f[i][j][k] += 1ll*c[n-l][i-l]*c[m-r][j-r]%mod*g[i-l][j-r][a[k]]%mod*f[l][r][k-1]%mod) %= mod;
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++) (ans += f[i][j][C]) %= mod;
cout << ans << endl;
return 0;
}

[洛谷P3158] [CQOI2011]放棋子的更多相关文章

  1. 洛谷P3158 [CQOI2011]放棋子 组合数学+DP

    题意:在一个m行n列的棋盘里放一些彩色的棋子,使得每个格子最多放一个棋子,且不同颜色的棋子不能在同一行或者同一列.有多少祌方法? 解法:这道题不会做,太菜了qwq.题解是看洛谷大佬的. 设C是组合数, ...

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

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

  3. 洛谷 P3182 [HAOI2016]放棋子(高精度,错排问题)

    传送门 解题思路 不会错排问题的请移步——错排问题 && 洛谷 P1595 信封问题 这一道题其实就是求对于每一行的每一个棋子都放在没有障碍的地方的方案数. 因为障碍是每行.每列只有一 ...

  4. 洛谷P3182 [HAOI2016]放棋子

    P3182 [HAOI2016]放棋子 题目描述 给你一个N*N的矩阵,每行有一个障碍,数据保证任意两个障碍不在同一行,任意两个障碍不在同一列,要求你在这个矩阵上放N枚棋子(障碍的位置不能放棋子),要 ...

  5. 题解 P3158 [CQOI2011]放棋子

    题解 本题是一个 \(DP\) 加 容斥,容斥的式子很好推,重点是如何想到和如何推出 \(DP\) 部分的式子. 因为不同种颜色的棋子不能放在同一行或同一列,所以不同种的棋子是相对独立的. 据此,我们 ...

  6. 洛谷 P3182 [HAOI2016]放棋子(错排问题)

    题面 luogu 题解 裸的错排问题 错排问题 百度百科:\(n\)个有序的元素应有\(n!\)个不同的排列,如若一个排列使得所有的元素不在原来的位置上,则称这个排列为错排:有的叫重排.如,1 2的错 ...

  7. P3158 [CQOI2011]放棋子

    传送门 题解(因为公式太多懒得自己抄写一遍了--) //minamoto #include<bits/stdc++.h> #define ll long long #define R re ...

  8. BZOJ 3294: [Cqoi2011]放棋子

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

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

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

随机推荐

  1. 团队作业2 <嗨,你的快递!>需求分析与原型设计

    哦,不,是你的快速(*_*) 第一部分 需求分析 1.1 用户调研 1.1.1调研对象:由于我们的系统是校园快递代取业务,面向的是大学生活,所以本次调研范围都是在校大学生(除了师大学生,也包括了外校的 ...

  2. UDP与TCP笔记

    1.UDP UDP协议在工作时是建立在IP协议之上的,UDP从进程的缓冲区接收进程每一次产生的输出,对每次输出都生成一个UDP数据报,然后把生成的UDP数据报直接封装在IP数据报中进行传输,因此在传输 ...

  3. 动态生成CheckBox(Winform程序)

    在做用户权限设置功能时,需要做一个动态生成权限列表的功能.(笔记.分享) //1.清空权限控件组的默认控件 panelPermissions.Controls.Clear(); _groupBoxLi ...

  4. typedef struct bit0 : 1

    这句话定义了一个位域,bit0是该位域的域名,而且bit0只占用一个位.位域是指信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位.为了节省存储空间,并使处理简便,C语言提供了一种 ...

  5. js访问对象属性的两种方法

    var obj={name:'fuuf',age:19} 第一种,用.访问 obj.name 第二种 用[]访问 obj['name']  //此时name是字符串,要加引号 注意事项 使用第二种方法 ...

  6. 爬虫学习之-scrapy交互式命令

    scrapy shell https:///www.baidu.com  会启动爬虫请求网页 view(response) 会在浏览器打开请求到的临时文件 response.xpath("/ ...

  7. [转帖] Oracle 关闭自动收集统计信息

    --关闭自动统计信息 https://blog.csdn.net/royzhang7/article/details/51172556 明天再仔细看一下. select client_name,sta ...

  8. SpringBoot(十二)_springboot整合PageHelper

    我之所以会发现这个PageHelper这个东东 是因为公司在使用 ,刚开始我也没太注意这个插件,感觉不就是个分页插件吗?也就那样,直到一天,我在网上找了个代码生成器,用来构建代码,因为它是针对mysq ...

  9. java异常处理-finally中使用return和throw语句

    java异常语句中的finally块通常用来做资源释放操作,如关闭文件.关闭网络连接.关闭数据库连接等.正常情况下finally语句中不应该使用return语句也不应该抛出异常,以下讨论仅限于java ...

  10. 栈java实现

    这几天,过得挺充实的,每天都在不停的上课,早上很早就起来去跑步,晚上到图书馆看书.一边紧张的学习,一边在默默的备战软考.最近还接手了一个公司官网的建设.这是我在川信最后的一个完整学期了,每件事我都要认 ...