[题解] [CQOI2011] 放棋子

题面

题解

为了练习计数而做

注意到一种颜色占据的行, 列其他的颜色不能放

又考虑到我们并不需要知道哪些行哪些列选了, 只需要知道还有几行几列没选即可

于是有 \(f[i][j][k]\) 代表前 \(i\) 种颜色选完之后, 还有 \(j\) 行没选, \(k\) 列没选的方案数

\(g[i][j][k]\) 代表, \(i\) 个棋子, 放在 \(j\) 行 \(k\) 列中并且没有空行空列的方案数

\(cnt_i\) 代表颜色为 \(i\) 的棋子有几个


\[
\displaystyle g[i][j][k] = \binom{j*k}{i}-\sum_{x=0}^{j}\sum_{y=0 \&\& x*y \geq i}^{k}(-1)^{x + y}\binom{j}{x}\binom{k}{y}\binom{x*y}{i}\\f[i][j][k] = \sum_{x=0}^{x+j \leq n}\sum_{y=0\&\&x*y\geq cnt_i}^{y+k\leq n}f[i - 1][j+x][k+y]\binom{j+x}{x}\binom{k+y}{y}g[x][y][cnt_i]
\]
这个简单容斥一下就可以了

然后, 就没了???

Code

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
const int N = 1005;
const int mod = 1000000009;
using namespace std;

int n, m, K, cnt[N], c[N][N], f[15][35][35], g[905][35][35], mx, ans; 

template < typename T >
inline T read()
{
    T x = 0, w = 1; char c = getchar();
    while(c < '0' || c > '9') { if(c == '-') w = -1; c = getchar(); }
    while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
    return x * w;
}

int main()
{
    n = read <int> (), m = read <int> (), K = read <int> ();
    for(int i = 1; i <= K; i++)
        mx = max(mx, cnt[i] = read <int> ());
    for(int i = 0; i <= n * m; i++)
        for(int j = 0; j <= i; j++)
            c[i][j] = (!j ? 1 : (c[i - 1][j - 1] + c[i - 1][j]) % mod);
    for(int i = 1; i <= mx; i++)
        for(int j = 0; j <= n; j++)
            for(int k = 0; k <= m; k++)
                for(int x = 0; x <= j; x++)
                    for(int y = 0; y <= k; y++)
                        if((j - x) * (k - y) >= i)
                            g[i][j][k] = (1ll * g[i][j][k] +
                                          1ll * c[j][x] * c[k][y] % mod
                                          * ((x + y) & 1 ? -1 : 1) * c[(j - x) * (k - y)][i] % mod + mod) % mod;
    f[0][n][m] = 1;
    for(int k = 1; k <= K; k++)
        for(int i = 0; i <= n; i++)
            for(int j = 0; j <= m; j++)
                for(int x = 0; x + i <= n; x++)
                    for(int y = 0; y + j <= m; y++)
                        if(f[k - 1][i + x][j + y] && x * y >= cnt[k])
                            f[k][i][j] = (1ll * f[k][i][j] +
                                          1ll * f[k - 1][i + x][j + y] * c[i + x][x] % mod
                                              * c[j + y][y] % mod * g[cnt[k]][x][y] % mod) % mod;
    for(int i = 0; i <= n; i++)
        for(int j = 0; j <= m; j++)
            ans = (1ll * ans + f[K][i][j]) % mod;
    printf("%d\n", ans);
    return 0;
}

原文地址:https://www.cnblogs.com/ztlztl/p/12199056.html

时间: 2024-09-30 00:35:09

[题解] [CQOI2011] 放棋子的相关文章

BZOJ 3294: [Cqoi2011]放棋子

3294: [Cqoi2011]放棋子 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 628  Solved: 238[Submit][Status][Discuss] Description Input 输入第一行为两个整数n, m, c,即行数.列数和棋子的颜色数.第二行包含c个正整数,即每个颜色的棋子数.所有颜色的棋子总数保证不超过nm. Output 输出仅一行,即方案总数除以 1,000,000,009的余数. Sample Input

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

P3158 [CQOI2011]放棋子 放棋子的顺序和方案数无关,所以可以从按颜色递推 设$f[u][p][k]$为放到第$u$种颜色,所剩空间$p*k$的方案数 $g[u][i][j]$表示第$u$种颜色占据$i*j$空间的方案数,可以预处理 $g[u][i][j]=\binom{i*j}{c[u]}-\sum_{p=1}^{i}\sum_{k=1}^{j}g[u][p][k]*\binom{i}{i-p}*\binom{j}{j-k}*[p<i||j<k]$ $f[u][p][k]=\su

[CQOI2011] 放棋子 - 计数dp

在一个 \(m\) 行 \(n\) 列的棋盘里放一些彩色的棋子,使得每个格子最多放一个棋子,且不同颜色的棋子不能在同一行或者同一列,有多少种方法? Solution 设 \(f[i][j][k]\) 表示用前 \(k\) 种颜色的棋子,占领了 \(i\) 行 \(j\) 列的方案数 设 \(g[i][j][k]\) 表示用任意 \(k\) 个同色棋子占领 \(i\) 行 \(j\) 列的方案数,则考虑总方案数 - 实际上有没有被占领的行或列的方案数,则 \[g[i][j][k]=C_{ij}^k

[CQOI2011]放棋子

想到了50%吧算是. f[i][j][k]表示,前i种,占了j行k列.方案数. 发现,转移要处理:"用c个棋子,占据n行m列"的方案数. 设g[i][j][k]表示,i行j列用k个棋子占的方案数.直接处理复杂度爆炸. 然后我就mengbier了. 考虑大力容斥: 也即,总方案数-不合法方案数(不能覆盖完全) g[i][j][k]=C(i*j,k)-∑l∑r:g[l][r][k]*C(i,l)*C(j,r) (i*j>=k&&l<=i&&j&l

bzoj4563[Haoi2016]放棋子

bzoj4563[Haoi2016]放棋子 题意: 给你一个N*N的矩阵,每行有一个障碍,数据保证任意两个障碍不在同一行,任意两个障碍不在同一列.要求你放N枚棋子(障碍的位置不能放棋子),也满足每行只有一枚棋子,每列只有一枚棋子的限制,求有多少种方案.N≤200. 题解: 发现在障碍在什么位置和答案无关.因此可以把棋子摆成左上-右下对角线,然后就是错排问题(一个长度为n的排列,要求每个元素不能放在与自己编号相同的位置上,问有多少种方案满足条件)了.错排公式:f[i]=(f[i-1]+f[i-2]

LibreOJ #2061. 「HAOI2016」放棋子

二次连通门 : LibreOJ #2061. 「HAOI2016」放棋子 /* LibreOJ #2061. 「HAOI2016」放棋子 MDZZ ... 错排 + 高精 */ #include <iostream> #include <cstdio> #include <vector> #include <iomanip> #include <cassert> #include <algorithm> #define int64 l

bzoj4563【HAOI2016】放棋子

4563: [Haoi2016]放棋子 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 172  Solved: 119 [Submit][Status][Discuss] Description 给你一个N*N的矩阵,每行有一个障碍,数据保证任意两个障碍不在同一行,任意两个障碍不在同一列,要求你在 这个矩阵上放N枚棋子(障碍的位置不能放棋子),要求你放N个棋子也满足每行只有一枚棋子,每列只有一枚棋子 的限制,求有多少种方案. Input 第一行

[BZOJ 4563][Haoi2016]放棋子(错排公式)

Description 给你一个N*N的矩阵,每行有一个障碍,数据保证任意两个障碍不在同一行,任意两个障碍不在同一列,要求你在这个矩阵上放N枚棋子(障碍的位置不能放棋子),要求你放N个棋子也满足每行只有一枚棋子,每列只有一枚棋子的限制,求有多少种方案. Solution 其实和给出的障碍没什么关系,可以直接上错排公式(因为一开始的障碍可以看做一个排列,然后放棋子等同于要你生成一个新的不能有和原来位置相同元素的排列) f[n]=(f[n-1]+f[n-2])*(n-1)(为什么的话上一篇里有说Qw

[HAOI2016]放棋子

题意 Here 思考 看第一眼:状压dp,再看范围gg 第二眼:普通dp,貌似可以直接递推? 其实就是个很裸的错排问题,写个博客顺便复习下~ 错排问题就是说一个 \(n\) 的排列,每个元素都满足 \(a[i] != i\),求方案数 记 \(f[n]\) 为 \(n\) 的错排方案数,我们可以考虑递推: 放第 \(n\) 个元素,有 \(n-1\) 种方法 假设第 \(n\) 个元素放在了 \(p\) 位置,那么放编号为 \(p\) 的元素: 放在 \(n\) 位置,对于剩下 \(n-2\)