Luogu P2051 [AHOI2009]中国象棋(dp)
题面
题目描述
这次小可可想解决的难题和中国象棋有关,在一个 \(N\) 行 \(M\) 列的棋盘上,让你放若干个炮(可以是 \(0\) 个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法。大家肯定很清楚,在中国象棋中炮的行走方式是:一个炮攻击到另一个炮,当且仅当它们在同一行或同一列中,且它们之间恰好 有一个棋子。你也来和小可可一起锻炼一下思维吧!
输入输出格式
输入格式:
一行包含两个整数 \(N\) , \(M\) ,之间由一个空格隔开。
输出格式:
总共的方案数,由于该值可能很大,只需给出方案数模 \(9999973\) 的结果。
输入输出样例
输入样例:
1 3
输出样例:
7
说明
样例说明
除了 \(3\) 个格子里都塞满了炮以外,其它方案都是可行的,所以一共有 \(2 \times 2 \times 2-1=7\) 种方案。
数据范围
$100 % $的数据中 \(N\) 和 \(M\) 均不超过 \(100\)
$50 % $的数据中 \(N\) 和 \(M\) 至少有一个数不超过 \(8\)
$30 % $ 的数据中 \(N\) 和 \(M\) 均不超过 \(6\)
思路
晚上帮 \(alec\) 大佬调 \(\LaTeX\) 结果还是没有调好,于是就来做题,没想到这个紫题这么简单。(奶一口,一个月之内它会变蓝)
很显然,同行同列不能摆超过两个炮,所以首先想到的是类似八皇后啥的的状压 \(DP\) ,但是本题要记录没有放棋子、放了一个棋子、放了两个棋子这三种状态,要用我不会打(其实是懒)的手写三进制,而且 \(3^{100}\) 内存也是不允许的,所以考虑用其它方法来更新。
定义 \(f[i][j][k]\) 表示放了前 \(i\) 行后共有 \(j\) 列上各有两个炮,且共有 \(k\) 列上各有一个炮的情况的方案数。首先,显然有 \(f[0][0][0]=1\) 。而对于新的一行,我们就可以这样更新:
- 这一行不放: \(f[i][j][k]+=f[i-1][j][k]\) ;
- 这一行放了一个,且这一个放在原有一个炮的一列上: \(f[i][j][k]+=f[i-1][j-1][k+1] \times (k+1)\) ;
- 这一行放了一个,且这一个放在没有炮的一列上: \(f[i][j][k]+=f[i-1][j][k-1] \times (m-j-k+1)\) ;
- 这一行放了两个,且这两个都放在原来没有炮的两列上: \(f[i][j][k]+=f[i-1][j][k-2] \times C_{m-j-k+2}^2\) ;
- 这一行放了两个,且一个放在原来没有炮的一列上,另一个放在原来有一个炮的一列上: \(f[i][j][k]+=f[i-1][j-1][k] \times (m-j-k+1) \times k\) ;
- 这一行放了两个,且两个都放在原来有一个炮的两列上: \(f[i][j][k]+=f[i-1][j-2][k+2] \times C_{k+2}^2\) ;
一共有 \(6\) 种转移途径呢!但是并不是每种状态都可以有六种转移方式,我们还要在代码中加入对于边界的判断,这样就能顺利 \(AC\) 了。
AC代码
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const LL P=9999973;
LL n,m,ans,f[105][105][105];
LL C(LL x){return x*(x-1)/2;}
int main()
{
scanf("%lld%lld",&n,&m);
f[0][0][0]=1;
for(LL i=1;i<=n;i++)
for(LL j=0;j<=m;j++)
for(LL k=0;k+j<=m;k++)
{
f[i][j][k]=f[i-1][j][k];///这一行不放
///放一个
if(j-1>=0&&k+1<=m) f[i][j][k]=(f[i][j][k]+f[i-1][j-1][k+1]*(k+1))%P;///从k变为j
if(k-1>=0) f[i][j][k]=(f[i][j][k]+f[i-1][j][k-1]*(m-j-k+1))%P;///新增一个k
///放两个
if(k-2>=0) f[i][j][k]=(f[i][j][k]+f[i-1][j][k-2]*C(m-j-k+2))%P;///新增两个k状态
if(j-1>=0&&k-1>=0) f[i][j][k]=(f[i][j][k]+f[i-1][j-1][k]*(m-j-k+1)*k)%P;///1个从k变为j,一个新增k
if(j-2>=0&&k+2<=m) f[i][j][k]=(f[i][j][k]+f[i-1][j-2][k+2]*C(k+2))%P;///2个从k变为j
}
for(LL i=0;i<=m;i++)
for(LL j=0;i+j<=m;j++)
ans=(ans+f[n][i][j])%P;
printf("%lld",ans);
return 0;
}
Luogu P2051 [AHOI2009]中国象棋(dp)的更多相关文章
- [Luogu P2051] [AHOI2009]中国象棋 (状压DP->网格DP)
题面 传送门:https://www.luogu.org/problemnew/show/P2051 Solution 看到这题,我们不妨先看一下数据范围 30pt:n,m<=6 显然搜索,直接 ...
- [P2051 [AHOI2009]中国象棋] DP
https://www.luogu.org/problemnew/show/P2051 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一 ...
- Luogu P2051[AHOI2009]中国象棋【dp】By cellur925
题目传送门 题目大意:给定一个$n*m$的棋盘,求放三个“炮”使它们不共行也不共列的方案数.($n,m$$<=100$) 这题主要是转移比较困难,因为情况比较多,所以需要冷静大胆细心地进行分情况 ...
- 洛谷P2051 [AHOI2009]中国象棋(dp)
题面 luogu 题解 \(50pts:\)显然是\(3\)进制状压\(dp\) \(100pts:\) 一行一行地考虑 \(f[i][j][k]\)表示前\(i\)行,有\(j\)列放了一个,有\( ...
- P2051 [AHOI2009]中国象棋——DP(我是谁,我在哪,为什么)
象棋,给你棋盘大小,然后放炮(炮的数量不限),不能让炮打到其他的炮,问方案数: 数据n,m<=200; 状态压缩似乎能做,但是我不会: 因为只要状态数,所以不必纠结每种状态的具体情况: 可以想出 ...
- luogu P2051 [AHOI2009]中国象棋
统计方案,果断 dp 注意到合法方案即为每一行,每一列的棋子数不超过2 设\(f_{i,j,k}\)表示放到第\(i\)行,有\(j\)列可以放2个,有\(k\)列可以放1个的方案 然后就随便讨论一下 ...
- Luogu 2051[AHOI2009]中国象棋 - DP
Description 在 $n * m$ 的格子上放若干个炮, 使得每个炮都不能攻击到其他炮 Solution 定义数组f[ i ][ j ][ k ] 表示到了第 i 行, 已经有2个炮的列数为 ...
- 洛谷 P2051 [AHOI2009]中国象棋 状态压缩思想DP
P2051 [AHOI2009]中国象棋 题意: 给定一个n*m的空棋盘,问合法放置任意多个炮有多少种情况.合法放置的意思是棋子炮不会相互打到. 思路: 这道题我们可以发现因为炮是隔一个棋子可以打出去 ...
- 洛谷 P2051 [AHOI2009]中国象棋 解题报告
P2051 [AHOI2009]中国象棋 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法. ...
随机推荐
- java_初始网络编程
/** * 网咯编程入门: * c/s结构:全称Client/Server结构,是指客户端和服务器结构.常见程序有qq.迅雷等如那件 * B/S结构:全称Browser/Server结构,是指浏览 ...
- linux sudo命令失败 提示sudo:/usr/bin/sudo 必须属于用户 ID 0(的用户)并且设置 setuid 位
sudo:/usr/bin/sudo 必须属于用户 ID 0(的用户)并且设置 setuid 位 一.前言 这是一个神奇的错误,缘由是因为有人将/usr/bin/sudo的权限改为777或其他. 解决 ...
- [Ceoi2016|BZOJ4936] Match
哈希+分治+stack 题目: 给你一个由小写字母组成的字符串s,要你构造一个字典序最小的(认为左括号的字典序比右括号小)合法的括号 序列与这个字符串匹配,字符串和括号序列匹配定义为:首先长度必须相等 ...
- csp-s模拟测试56Merchant, Equation,Rectangle题解
题面:https://www.cnblogs.com/Juve/articles/11619002.html merchant: 二分答案,贪心选前m大的 但是用sort复杂度不优,会T掉 我们只是找 ...
- JAVA 垃圾回收读书笔记
对象已死 在JAVA代码运行中,会不停的创建对象,因为内存空间不是无限的,Java虚拟机必须不停的回收无用的数据空间.那么虚拟机是怎么判断对象空间是需要被回收的呢,也就是怎么样的数据算是垃圾数据呢? ...
- 12DUILib经典教程(实例)
Duilib经典实例教程:1基本框架:一个简单的Duilib程序一般是下面这个样子的:://Duilib使用设置部分:#pragmaonce:#defineWIN32_LEAN_AND_ME:#def ...
- 19-10-30-C
交文件吼啊. ZJ一下: T1是真·高中数学. T2不是很清楚,只得了30. T3打了一个欧拉序. 做的海星的地方: Vim太好用辣,直接按平常打叫上去它就是 freopen T1仔仔细细的研究了高考 ...
- 最新MySQL5.7.3.0版本安装(图文版本)
转自:http://www.myexception.cn/mysql/1609536.html 可以从http://dev.mysql.com/downloads/下载MySQL服务器安装软件包,我下 ...
- 解决MySQL登录ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using passwor)问题
问题描述 今天在MAC上安装完MySQL后,MYSQL默认给分配了一个默认密码,但当自己在终端上使用默认密码登录的时候,总会提示一个授权失败的错误:Access denied for user ‘ro ...
- PAT甲级——A1081 Rational Sum
Given N rational numbers in the form numerator/denominator, you are supposed to calculate their sum. ...