题目描述

这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法。大家肯定很清楚,在中国象棋中炮的行走方式是:一个炮攻击到另一个炮,当且仅当它们在同一行或同一列中,且它们之间恰好 有一个棋子。你也来和小可可一起锻炼一下思维吧!

输入输出格式

输入格式:

一行包含两个整数N,M,之间由一个空格隔开。

输出格式:

总共的方案数,由于该值可能很大,只需给出方案数模9999973的结果。

输入输出样例

输入样例#1:

1 3
输出样例#1:

7

说明

样例说明

除了3个格子里都塞满了炮以外,其它方案都是可行的,所以一共有2*2*2-1=7种方案。

数据范围

100%的数据中N和M均不超过100

50%的数据中N和M至少有一个数不超过8

30%的数据中N和M均不超过6

留给读者一点思考时间吧!

接下来,我来讲讲我怎么想的吧!

首先,做题之前,我们要冷静,不要看到省选题就 想AC 怕!

留心观察数巨范围,我们发现,这大概是标准的2-3维的动态规划题的数据规模。

既然这么想,那么我们肯定先考虑高维的状态设计了。

我是这么设计的:

首先它有n行m列,而两个炮又不能在一列,所以这么定义$f[i][j][k]$.

i表示已经放了i行棋子,作为第一维来枚举。

j表示在m列里,有j列只有一个炮。

最后,k表示有k列有两个炮。那么状态设计好了,怎么转移呢?

肯定的,我们先来枚举i行。

在这i行里,我们下棋的方案数:

1.首先肯定先要继承上一行枚举完的所有方案数,所以$$f[i][j][k]=f[i-1][j][k]$$

2.从最简单的下起,我们先一行只下一个吧,那么先找到空的行,没有棋子,我们可以随便怎么下。

那么有:$$f[i][j][k]+=f[i-1][j-1][k]*(m-j-k+1)$$

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]*C(m-j-k+2,2)$$

5.我们还可以下在两个原来都有一个炮的那两列:$$f[i][j][k]+=f[i-1][j+2][k-2]*C(j+2)$$

6.最后,其实还有一种下法,我们可以将一个炮下在没有炮的那一列,另一个下在有炮的一列。

那么有:$$f[i][j][k]+=f[i-1][j][k-1]*j*(m-j-k+1)$$

P.S. 为了不让数组越界,我们要加一些判断,如$if (j>=1)$之类的。

代码在下面啦

#include <bits/stdc++.h>
#define C(x) ((x)*(x-1)/2)
using namespace std; int main()
{
int n,m,ans=,mo=;
long long f[][][]={};
cin>>n>>m;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
for (int k=;k+j<=m;k++) {
f[i][j][k]=f[i-][j][k];
if (j>=)
f[i][j][k]+=f[i-][j-][k]*(m-k-j+),f[i][j][k]%=mo;
if (k>=)
f[i][j][k]+=f[i-][j+][k-]*(j+),f[i][j][k]%=mo;
if (j>=)
f[i][j][k]+=f[i-][j-][k]*C(m-j-k+),f[i][j][k]%=mo;
if (k>=)
f[i][j][k]+=f[i-][j][k-]*j*(m-k-j+),f[i][j][k]%=mo;
if (k>=)
f[i][j][k]+=f[i-][j+][k-]*C(j+),f[i][j][k]%=mo;
}
for (int j=;j<=m;j++)
for (int k=;k+j<=m;k++)
ans+=f[n][j][k],ans%=mo;
cout<<ans<<endl;
return ;
}

洛谷 P2051 [AHOI2009]中国象棋的更多相关文章

  1. 洛谷 P2051 [AHOI2009]中国象棋 解题报告

    P2051 [AHOI2009]中国象棋 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法. ...

  2. 洛谷 P2051 [AHOI2009]中国象棋 状态压缩思想DP

    P2051 [AHOI2009]中国象棋 题意: 给定一个n*m的空棋盘,问合法放置任意多个炮有多少种情况.合法放置的意思是棋子炮不会相互打到. 思路: 这道题我们可以发现因为炮是隔一个棋子可以打出去 ...

  3. [洛谷P2051] [AHOI2009]中国象棋

    洛谷题目链接:[AHOI2009]中国象棋 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法 ...

  4. 洛谷P2051 [AHOI2009]中国象棋(dp)

    题面 luogu 题解 \(50pts:\)显然是\(3\)进制状压\(dp\) \(100pts:\) 一行一行地考虑 \(f[i][j][k]\)表示前\(i\)行,有\(j\)列放了一个,有\( ...

  5. 洛谷P2051 [AHOI2009] 中国象棋(状压dp)

    题目简介 n*m的棋盘,对每行放炮,要求每行每列炮数<=2,求方案数%9999973 N,M<=100 题目分析 算法考虑 考虑到N,M范围较小,每一行状态只与前面的行状态有关,考虑状压D ...

  6. BZOJ1801或洛谷2051 [AHOI2009]中国象棋

    BZOJ原题链接 洛谷原题链接 这题挺难想状态的,刚看题感觉是状压,但数据\(100\)显然不可能. 注意到每行每列只能放\(0\sim 2\)个棋子,所以我们可以将这个写入状态. 设\(f[i][j ...

  7. 洛谷.2051.[AHOI2009]中国象棋(DP)

    题目链接 /* 每行每列不能超过2个棋子,求方案数 前面行对后面行的影响只有 放了0个.1个.2个 棋子的列数,与排列方式无关 所以设f[i][j][k]表示前i行,放了0个棋子的有j列,放了1个棋子 ...

  8. 洛谷2051 [AHOI2009]中国象棋

    题目链接 题意概述:n行m列棋盘放若干个棋子每行每列最多两个求方案总数,答案对9999973取模. 可以比较容易看出这是个dp,设f[i][j][k]表示前i行j列放1个棋子k列放2个棋子的方案总数. ...

  9. Luogu P2051 [AHOI2009]中国象棋(dp)

    P2051 [AHOI2009]中国象棋 题面 题目描述 这次小可可想解决的难题和中国象棋有关,在一个 \(N\) 行 \(M\) 列的棋盘上,让你放若干个炮(可以是 \(0\) 个),使得没有一个炮 ...

随机推荐

  1. 【共享单车】—— React后台管理系统开发手记:AntD Table基础表格

    前言:以下内容基于React全家桶+AntD实战课程的学习实践过程记录.最终成果github地址:https://github.com/66Web/react-antd-manager,欢迎star. ...

  2. [Angular] @ViewChild read custom directive and exportAs

    For example we have a component: <Card ></Card> And a driective: <Card highlighted> ...

  3. Android源代码下载

    清华大学AOSP镜像: https://mirrors.tuna.tsinghua.edu.cn/help/AOSP/

  4. 倍福TwinCAT(贝福Beckhoff)基础教程2.1 TwinCAT常见类型简介

    常见数据类型可以参考EXCEL表格的内容,在准备值中输入非法数据,然后回车会有错误提示(例如BYTE数据输入256)   右击可以切换二进制或十进制显示,wData5可以被挨个采集每个位上的数据,也可 ...

  5. 【Excle数据透视表】如何为数据透视表应用样式

    如下数据透视表样例,如何为该数据透视表设置样式呢? 步骤 单击数据透视表区域的任意单元格→数据透视表工具→设计→数据透视表样式→打开下拉箭头即可任意选择

  6. 【VBA】设置Excle最近使用文件清单数量

    打开Excle,点击"文件"------"最近使用的文件",如下图: 根据上图可以看到,最近使用的文件数目为11个,那么是怎么实现的呢?具体代码如下: Publ ...

  7. JAVA静态导入(inport static)详解

    在Java 5中,import语句得到了增强,以便提供甚至更加强大的减少击键次数功能,虽然一些人争议说这是以可读性为代价的.这种新的特性成为静态导入. 当你想使用static成员时,可以使用静态导入( ...

  8. 使用php在服务器端生成图文验证码

    图文验证码的实现原理: 1):准备些许图片将其存储在数据库,每一张图片对应一个标识字段. 2):在服务器端使用数组的形式将图片与标识字段组合起来. 3):随机给客户端返回图片,并接受用户输入的字段. ...

  9. A股市场底部顶部历史数据

    1. A股市场平均市盈率 大顶沪市平均市盈率:66-70倍. A股市场2次大底沪市平均市盈率:12倍-15倍. 大底时的例子. 2005年6月6日上证指数1000点时的14倍市盈率.2008年10月2 ...

  10. linux rm -rf * 文件恢复记

    手太快,肠子都毁清了.本来是删除一个文件 rm path/myfile.txt结果不知为何加了个*,变成了rm path/myfile.txt *赶紧ls,发现所有代码都化为了乌有,还没提交,还没备份 ...