国际象棋:模板棋盘状压。

摆上马:需要点思维的棋盘状压,相比上一道题加了“蹩马脚”的设定。

Easy_version :国际象棋

概述一下此类棋盘问题的思路:

  1. 用二进制数表示出棋盘上某一行的状态。
  2. 用位运算预处理出合法的单行状态,以及需要用到的一些东西。
  3. 用位运算判断前一行或者前几行能否转移过来。
  4. 转移前一行或者前几行的兼容类。

基础位运算

判断两边是否有:i&(i>>1)i&(i<<1)

判断 \(i\) 是否包含在合法状态 \(j\) 里: (i&j)==i

对于本题

记录下两行的状态,定义 \(dp[i][j][k][l]\) 表示第 \(i\) 行时,第 \(i\)行状态为 \(j\) ,第 \(i-1\) 行状态为 \(k\) ,且目前摆了 \(l\) 个马的方案数。

初始化 \(dp[0][0][0][0]=1\) 。

循环顺序是:

  1. 行数。
  2. 马的个数。
  3. 第 \(i\) 行状态。
  4. 第 \(i-1\) 行状态。
  5. 第 \(i-2\) 行状态。

注意由于只会用到前一行的状态,所以我们可以强行滚动数组优化,通过 \(\bmod 2\) 来解决。滚动数组每次必须先初始化为 \(0\) 。

时间复杂度 \(O(2^{3n}mk)\),有点紧,常数不能太大。

为了不在最后统计一遍,太过麻烦,所以我们多加了两行,并强制这两行必须不能放马,这样就不用手工统计方案数了。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> pi;
int n,m,k,tot[70];
const ll mod=1e9+7;
ll dp[2][70][70][25];
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
cin>>n>>m>>k;
for(int i=0;i<(1<<n);i++)
{
for(int j=0;j<n;j++)
{
tot[i]+=((i>>j)&1);
}
}
dp[0][0][0][0]=1;
for(int i=1;i<=m+2;i++)
{
for(int j=0;j<=k;j++)
{
for(int a=0;a<(1<<n);a++)
{
for(int b=0;b<(1<<n);b++)
{
dp[i&1][a][b][j]=0;
if(((a>>2)&b)||((a<<2)&b))continue;
for(int c=0;c<(1<<n);c++)
{
if(((a>>1)&c)||((a<<1)&c))continue;
if(tot[a]<=j)
{
dp[i&1][a][b][j]=(dp[i&1][a][b][j]+dp[(i&1)^1][b][c][j-tot[a]])%mod;
}
}
}
}
}
}
cout<<dp[(m+2)&1][0][0][k];
return 0;
}

Hard_version :中国象棋 - 摆上马

相比上一道,这道必须考虑蹩马脚的情况,且不用记录马的个数。

这题难点便是在判断蹩马脚:

进阶位运算

筛选出本行左边是 \(0\) ,这一位是 \(1\) 的点: i&((~i)>>1)

筛选出本行右边是 \(0\) ,这一位是 \(1\) 的点: i&((~i)<<1)

考虑 \(i\) 和 \(i-1\) 行时

\(a\) 为第 \(i\) 行状态,\(b\) 为第 \(i-1\) 行状态,\(c\) 为第 \(i-2\) 行状态。

第 \(i\) 行左边没有蹩马脚,并且可以攻击到左上的马的情况:a&((~a)>>1)&(b>>2)

第 \(i\) 行右边没有蹩马脚,并且可以攻击到右上的马的情况:a&((~a)<<1)&(b<<2)

第 \(i-1\) 行左边没有蹩马脚,并且可以攻击到左下的马的情况:b&((~b)>>1)&(a>>2)

第 \(i-1\) 行右边没有蹩马脚,并且可以攻击到右下的马的情况:b&((~b)<<1)&(a<<2)

考虑 \(i\) 和 \(i-2\) 行时

第 \(i\) 行上边没有蹩马脚,并且可以攻击到左上的马的情况:a&(~b)&(c>>1)

第 \(i\) 行上边没有蹩马脚,并且可以攻击到右上的马的情况:a&(~b)&(c<<1)

第 \(i-2\) 行下边没有蹩马脚,并且可以攻击到左下的马的情况:c&(~b)&(a>>1)

第 \(i-2\) 行下边没有蹩马脚,并且可以攻击到右下的马的情况:c&(~b)&(a<<1)

时间复杂度 \(O(2^{3y}x)\) 。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> pi;
int n,m;
const ll mod=1e9+7;
ll dp[2][70][70];
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
cin>>m>>n;
dp[0][0][0]=1;
for(int i=1;i<=m+2;i++)
{
for(int a=0;a<(1<<n);a++)
{
for(int b=0;b<(1<<n);b++)
{
dp[i&1][a][b]=0;
if((a&((~a)>>1)&(b>>2))||((a&((~a)<<1)&(b<<2)))||(b&((~b)>>1)&(a>>2))||(b&((~b)<<1)&(a<<2)))continue;
for(int c=0;c<(1<<n);c++)
{
if(((~b)&a&(c>>1))||((~b)&a&(c<<1))||((~b)&c&(a<<1))||((~b)&c&(a>>1)))continue;
dp[i&1][a][b]=(dp[i&1][a][b]+dp[(i&1)^1][b][c])%mod;
}
}
}
}
cout<<dp[(m+2)&1][0][0];
return 0;
}

Luogu P5005 中国象棋 - 摆上马 / Luogu P8756 国际象棋 题解 [ 蓝 ] [ 状压 dp ] [ 位运算 ]的更多相关文章

  1. BZOJ 4042 Luogu P4757 [CERC2014]Parades (树形DP、状压DP)

    题目链接 (BZOJ) https://www.lydsy.com/JudgeOnline/problem.php?id=4042 (Luogu) https://www.luogu.org/prob ...

  2. [Luogu P3959] 宝藏 (状压DP+枚举子集)

    题面 传送门:https://www.luogu.org/problemnew/show/P3959 Solution 这道题的是一道很巧妙的状压DP题. 首先,看到数据范围,应该状压DP没错了. 根 ...

  3. [Luogu P2831] 愤怒的小鸟 (状压DP)

    题面: 传送门:https://www.luogu.org/problemnew/show/P2831 Solution 首先,我们可以先康一康题目的数据范围:n<=18,应该是状压或者是搜索. ...

  4. FZU 1025 状压dp 摆砖块

    云峰菌曾经提到过的黄老师过去讲课时的摆砖块 那时百度了一下题目 想了想并没有想好怎么dp 就扔了 这两天想补动态规划知识 就去FZU做专题 然后又碰到了 就认真的想并且去做了 dp思想都在代码注释里 ...

  5. 状压DP之中国象棋

    题目 传送们 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可以是0个),使得没有一个炮可以攻击到另一个炮,请问有多少种放置方法.大家肯定很清楚,在中国象棋中炮的行走方 ...

  6. 状压dp:luogu P2704 [NOI2001]炮兵阵地

    https://www.luogu.org/problemnew/show/P2704 知识点:1.滚动数组:取模实现 2.位运算优先级最低 顾是if(!(a&b))而不是if(!a& ...

  7. luogu 2051 中国象棋

    非常好的dp,锻炼思维 f[i][j][k] 前i行有j列放1,k列放2 #include<bits/stdc++.h> #define int long long #define rep ...

  8. BZOJ 4006 Luogu P3264 [JLOI2015]管道连接 (斯坦纳树、状压DP)

    题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=4006 (luogu)https://www.luogu.org/probl ...

  9. luogu 2157 状压dp

    f[i][j][k]分别代表1-i-1个人全部打完饭时i及其后7个人的状态为j时最后一个打饭的人为i+k的状态下所用的最小时间 当i已经打过饭时 即 j&1 那么 f [i] [j>&g ...

  10. luogu 2704 炮兵阵地 状压dp

    状压的基础题吧 第一次看感觉难上天,后来嘛就.. 套路:先根据自身状态筛出可行状态,再根据地图等其他限制条件筛选适合的状态加入答案 f i,j,k 分别代表 行数,本行状态,上行状态,再累加答案即可 ...

随机推荐

  1. JDBC【4】-- SPI底层原理解析

    前面已经讲过SPI的基本实现原理了,demo也基本实现了,再来说说SPI. http://aphysia.cn/archives/jdbcspi 背景:SPI是什么? SPI,即是Service Pr ...

  2. 强网杯2023 谍影重重3.0 wp

    参考文章:[使用主动探测方法识别 U2hhZG93c29ja3M=(base64) 服务 - Phuker's Blog]:https://phuker.github.io/posts/U2hhZG9 ...

  3. uniapp+vue3酒店预订|vite5+uniapp预约订房系统模板(h5+小程序+App端)

    自研uni-app+vue3+uv-ui跨三端仿同程/携程酒店预订系统Uni-Vue3-WeTrip. uniapp-vue3-wetrip原创基于vite5+uniapp+pinia2+uni-ui ...

  4. Linx操作Nginx命令

    在 CentOS 上操作 Nginx 包括安装.启动.停止.重新加载配置等.以下是在 CentOS 上操作 Nginx 的常用命令: 安装 Nginx: sudo yum install nginx ...

  5. 不求甚解--详解ansible-playbook中roles的用法(二)

    前言 本文将详细介绍ansible-playbook中roles的各种用法 环境准备 组件 版本 操作系统 Ubuntu 22.04.4 LTS ansible 2.17.6 基本用法 文件结构 . ...

  6. 人工智能应用的“繁花时代”,各大企业何以破局AI模型挑战

    ​ AI技术的崛起,为各行业发展带来巨大变革和超强的创新潜力.然而,各大企业在拥抱AI的进程中并非一路坦途,"繁花盛开"的背后隐藏着AI模型生产与管理环节的诸多痛点. 先来看看部分 ...

  7. Mapstruct使用时报Unknown property xxx in result type xxx. Did you mean null

    0.背景 使用mapstruct时出现: Unknown property "xxx" in result type xxx. Did you mean "null&qu ...

  8. PostgreSQL 的特点

    title: PostgreSQL 的特点 date: 2024/12/24 updated: 2024/12/24 author: cmdragon excerpt: PostgreSQL 是当今最 ...

  9. Qt/C++音视频开发61-多屏渲染/一个解码渲染到多个窗口/画面实时同步

    一.前言 多屏渲染就是一个解码线程对应多个渲染界面,通过addrender这种方式添加多个绘制窗体,我们经常可以在展会或者卖电视机的地方可以看到很多电视播放的同一个画面,原理应该类似,一个地方负责打开 ...

  10. Qt音视频开发29-ffmpeg中x264/x265编码库支持

    一.前言 有了解码当然对应又有编码,编码是信息从一种形式或格式转换为另一种形式的过程也称为计算机编程语言的代码简称编码.用预先规定的方法将文字.数字或其它对象编成数码,或将信息.数据转换成规定的电脉冲 ...