bzoj 1081 [Ahoi2009] chess 中国象棋

题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1801

状态比较难设,的确没想到.

不关心第几列出现是否出现了棋子的个数.而是看看上一行第几列出现了1或2个棋子

利用组合巧妙解决问题.

设\(f[i][j][k]\)表示第i行有j列有1个棋子.有k行有2个棋子.

那么为空的即为\(m - j - k\)

状态转移方程:

1.这一行什么都不放:

\(f[i][j][k] += f[i - 1][j][k]\)

2.这一行放一个 在空行上放

\(f[i][j][k] += f[i - 1][j - 1][k] * (m - (j - 1) - k);\)

3.这一行放一个 在有一个棋子放

\(f[i][j][k] += f[i - 1][j + 1][k - 1] * (j + 1);\)

4.这一行放两个 都在一个棋子上放

\(f[i][j][k] += f[i - 1][j + 2][k - 2] * C(j + 2,2);\)

5.这一行放两个 在没有棋子上面放

\(f[i][j][k] += f[i - 1][j - 2][k] * C(m - j - k + 2,2);\)

6.这一行一个棋子在一个棋子的列上放,一个棋子在没有棋子的列上方

\(f[i][j][k] += f[i - 1][j][k - 1] * j * (m - j - k + 1)\)

边界的话:当然是\(f[0][0][0] = 1\)啦

滚动一下数组非常快

/*
卡常记录 :
总耗时 : 111ms -> 86ms
最高用时 : 23ms -> 17ms
*/
#include <iostream>
#include <cstdio>
#define rep(i,x,p) for(register int i = x;i <= p;++ i)
#define sep(i,x,p) for(register int i = x;i >= p;-- i)
#define gc getchar()
#define pc putchar
const int maxN = 100 + 7;
const int mod = 9999973; long long f[2][maxN][maxN];
int n,m; inline int read() {
int x = 0,f = 1;char c = gc;
while(c < '0' || c > '9') {if(c == '-')f = -1;c = gc;}
while(c >= '0' && c <= '9') {x = x * 10 + c - '0';c = gc;}
return x * f;
} inline int C(int n) {
return n * ( n - 1 ) / 2;
} int main() {
f[0][0][0] = 1;
int n,m;
n = read();m = read();
rep(i,1,n) {
int x = i % 2,q = x ? 0 : 1;
rep(j , 0 , m) {
for(register int k = 0;k + j <= m;++ k) {
f[x][j][k] = f[q][j][k];
if(j >= 1) f[x][j][k] += f[q][j - 1][k] * (m - (j - 1) - k);
if(k >= 1) f[x][j][k] += f[q][j + 1][k - 1] * (j + 1);
if(k >= 2) f[x][j][k] += f[q][j + 2][k - 2] * C(j + 2);
if(j >= 2) f[x][j][k] += f[q][j - 2][k] * C(m - j - k + 2);
if(k >= 1) f[x][j][k] += f[q][j][k - 1] * j * (m - j - k + 1);
f[x][j][k] %= mod;
}
}
}
long long ans = 0,x = n % 2;
rep(j , 0 , m) {
for(register int k = 0;k + j <= m;++ k)
ans += f[x][j][k];
ans %= mod;
}
printf("%lld\n", ans);
return 0;
}

Bzoj 1081 [Ahoi2009] chess 中国象棋的更多相关文章

  1. BZOJ 1801: [Ahoi2009]chess 中国象棋( dp )

    dp(i, j, k)表示考虑了前i行, 放了0个炮的有j列, 放了1个炮的有k列. 时间复杂度O(NM^2) -------------------------------------------- ...

  2. bzoj 1801: [Ahoi2009]chess 中国象棋

    Description 在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮. 请问有多少种放置方法,中国像棋中炮的行走方式大家应该很清楚吧. Input 一行包含两个整数N, ...

  3. [BZOJ 1801] [Ahoi2009]chess 中国象棋 【DP】

    题目链接:BZOJ - 1801 题目分析 对于50%的数据是可以直接状压 DP 的. 对于100%的数据,使用递推的 DP .(或者这只叫递推不叫 DP ?) 可以发现,每一行和每一列的棋子个数不能 ...

  4. BZOJ 1801: [Ahoi2009]chess 中国象棋 [DP 组合计数]

    http://www.lydsy.com/JudgeOnline/problem.php?id=1801 在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮. 请问有多少种放 ...

  5. bzoj 1801: [Ahoi2009]chess 中国象棋【dp】

    注意到一行只能放012个炮,我们只需要知道列的状态,不用状压行 所以设f[i][j][k]表示前i行有j列有1个炮,有k列有2个炮的方案数 然后分情况讨论转移就行了 #include<cstdi ...

  6. BZOJ_1801_[Ahoi2009]chess 中国象棋_DP

    BZOJ_1801_[Ahoi2009]chess 中国象棋_DP Description 在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮. 请问有多少种放置方法,中国像 ...

  7. 【BZOJ1801】[Ahoi2009]chess 中国象棋 DP

    [BZOJ1801][Ahoi2009]chess 中国象棋 Description 在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮. 请问有多少种放置方法,中国像棋中炮 ...

  8. BZOJ1801 Ahoi2009 chess 中国象棋 【DP+组合计数】*

    BZOJ1801 Ahoi2009 chess 中国象棋 Description 在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮. 请问有多少种放置方法,中国像棋中炮的行 ...

  9. BZOJ1801 [Ahoi2009]chess 中国象棋(DP, 计数)

    题目链接 [Ahoi2009]chess 中国象棋 设$f[i][j][k]$为前i行,$j$列放了1个棋子,$k$列放了2个棋子的方案数 分6种情况讨论,依次状态转移. #include <b ...

随机推荐

  1. 3、kvm配置vnc

    配置kvm通过vnc访问 virsh edit privi-server 添加如下配置: <graphics type='vnc' port='5901' autoport='no' liste ...

  2. UVa 10256(凸包、线段交、点在多边形内)

    要点 红蓝点分别求凸包显然 判断两凸包是否相交方法:所有红点不在蓝凸包内,反之亦然:所有红凸包线不与蓝凸包线相交,反之亦然. 书上让特判一下凸包退化成点或线段的情况,为什么我感觉代码仓库的代码并没特判 ...

  3. 【ACM】阶乘之和 - 避免重复计算

    阶乘之和 时间限制:3000 ms  |  内存限制:65535 KB 难度:3   描述 给你一个非负数整数n,判断n是不是一些数(这些数不允许重复使用,且为正数)的阶乘之和,如9=1!+2!+3! ...

  4. adb 调系统时间

    1.修改前提 获取系统root权限,然后adb shell进入shell界面 adb shell su 2.时区设置 cat /data/property/persist.sys.timezone / ...

  5. Java面向对象_常用类库api——日期操作类

    Data类 类Data表示特定的瞬间,精确到毫秒,也就是程序运行时的当前时间 Data data=new Data();//实例化Data对象,表示当前时间 Calendar类 日历类,使用此类可以将 ...

  6. Linux中ext2文件系统的结构

    1.ext2产生的历史 最早的Linux内核是从MINIX系统过渡发展而来的.Linux最早的文件系统就是MINIX文件系统.MINIX文件系统几乎到处都是bug,采用的是16bit偏移量,最大容量为 ...

  7. 【Linux】Linux查看程序端口占用情况

    使用命令查询8880端口的占用信息: netstat -naop|grep 查询结果: 发现8880端口被PID为4518的进程占用 使用命令查询所有的进程和端口使用情况: netstat –apn ...

  8. Nginx负载均衡与反向代理的配置实例

    user www www; worker_processes 10; error_log /data1/logs/nginx_error.log crit; pid /usr/local/webser ...

  9. Mercurial(HG) Windows+Eclipse安装、配置、使用

    Mercurial(HG) Windows客户端安装 Mercurial(HG): http://mercurial.selenic.com/ Windows客户端下载:http://mercuria ...

  10. C#获得字符串首字符字母(大写)

    /// <summary> /// 获得字符串首字符字母(大写): /// </summary> /// <param name="cnChar"&g ...