题目传送门

题目大意:给定一个$n*m$的棋盘,求放三个“炮”使它们不共行也不共列的方案数。($n,m$$<=100$)


这题主要是转移比较困难,因为情况比较多,所以需要冷静大胆细心地进行分情况讨论。

首先我们还是设计出状态:设$f[i][j][k]$表示前$i$行,放1枚棋子的有$j$列,放2枚棋子的有$k$列的方案数。

我们这样思考:放几个?放在哪?

  • 在第$i$行不放棋子。显然我们可以由$f[i-1][j][k]$转移过来。
(f[i][j][k]+=f[i-][j][k])%=moder;
  • 在第$i$行放1个棋子。有两个位置可以选择(放1个棋子的列,没放过棋子的列)

    • 放在之前有一个棋子的列,那么有一个棋子的列数变少,有两个棋子的列数变多。那么我们回到之前的状态,可以从$f[i-1][j+1][k-1]$转移来,而根据容斥的思想,我们有$(j+1)$个列可供选择。
if(k->=) (f[i][j][k]+=f[i-][j+][k-]*(j+))%=moder;
    • 放在之前没有棋子的列,那么有一个棋子的列数变多,之前可转移来的状态是$f[i-1][j-1][k]$。同理,我们有$(m-(j-1)-k)$个位置可以选择。
if(j->=) (f[i][j][k]+=f[i-][j-][k]*(m-k-j+))%=moder;
  • 在第$i$行放2个棋子。

    • 两个都放在不相同的没有棋子的列,那么有一个棋子的列数变多。之前可转移来的状态是$f[i-1][j-2][k]$。在空的列数中选2个,用到了组合数。
if(j->=) (f[i][j][k]+=f[i-][j-][k]*C(m-k-j+))%=moder;
    • 两个都放在不相同的已有一个棋子的列,那么有一个棋子的列数变少,有两个棋子的列数变多。之前可转移来的状态是$f[i-1][j+2][k-2]$。同样要用到组合数。
if(k->=) (f[i][j][k]+=f[i-][j+][k-]*C(j+))%=moder;
    • 两个棋子,一个放在已有一个棋子的列,一个放在没有棋子的列,那么有一个棋子的列数减一再加一相当于没变,有两个棋子的列数增多。并运用乘法原理。
if(k->=) (f[i][j][k]+=f[i-][j][k-]*j*(m-j-k+))%=moder;

列出了转移方程,我们的代码也就写完了(雾)。

Code

 #include<cstdio>
#include<algorithm> using namespace std;
typedef long long ll;
const ll moder=; int n,m;
ll ans,f[][][]; ll C(int x)
{
return (x*(x-))>>;
} int main()
{
scanf("%d%d",&n,&m);
f[][][]=;
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
for(int k=;k<=m-j;k++)
{
(f[i][j][k]+=f[i-][j][k])%=moder;
if(k->=) (f[i][j][k]+=f[i-][j+][k-]*(j+))%=moder;
if(j->=) (f[i][j][k]+=f[i-][j-][k]*(m-k-j+))%=moder;
if(k->=) (f[i][j][k]+=f[i-][j][k-]*j*(m-j-k+))%=moder;
if(k->=) (f[i][j][k]+=f[i-][j+][k-]*C(j+))%=moder;
if(j->=) (f[i][j][k]+=f[i-][j-][k]*C(m-k-j+))%=moder;
}
for(int j=;j<=m;j++)
for(int k=;k<=m-j;k++)
(ans+=f[n][j][k])%=moder;
printf("%lld",ans);
return ;
}

转移的时候我竟然想,为什么没有“两个棋子放在同一个之前没放到的列”这种情况。后来才意识到,我们每次面对的,是一行,其实是一个向量,(一维数组)。每一列只能放一颗棋子...

分类讨论大法好!

最新文章

  1. CodeChef - QCHEF 分块
  2. rplidar测试
  3. mysql my.cnf 配置详解
  4. 在WordPress后台菜单系统中添加Home链接
  5. JSON详解 .net
  6. 采用FLAG_ACTIVITY_CLEAR_TOP退出 多activity 或 整个程序
  7. UIBezierPathStudyDemo
  8. 深入探讨MFC消息循环和消息泵
  9. 怎样将IPhone应用程序软件公布到应用程序商店?
  10. ARP欺骗分析
  11. xcode修改代码目录结构出现clang:error:nosuchfileordirectory解决方法
  12. x264源代码简单分析:x264命令行工具(x264.exe)
  13. vs2019 cdkey 秘钥
  14. Reading comprehension HDU - 4990
  15. [转帖]Asp.Net MVC EF各版本区别
  16. 【PyQt5-Qt Designer】制作炫酷的启动界面+进度条
  17. manifest.xml微擎系统模块的安装文件内容
  18. 有趣的background
  19. 织梦SQL标签的使用
  20. 详解Android微信登录与分享

热门文章

  1. java基础知识汇总6(html篇)
  2. Android用户界面设计:基本button
  3. JavaScript 刚開始学习的人应知的 24 条最佳实践
  4. 【前端JS】radio 可单选可点击取消选中
  5. Django-权限信息中间件操作
  6. Linux查看IP 网关 DNS
  7. Python开发【深浅拷贝】
  8. 在Android Studio中修改应用包名
  9. SimpleAdapter ArrayAdapter用法
  10. ap和路由器有什么区别 ap和路由器的区别介绍【图文】