P5056 【模板】插头dp
\(\color{#0066ff}{ 题目描述 }\)
给出n*m的方格,有些格子不能铺线,其它格子必须铺,形成一个闭合回路。问有多少种铺法?
\(\color{#0066ff}{输入格式}\)
第1行,n,m(2<=n,m<=12)
从第2行到第n+1行,每行一段字符串(m个字符),"*"表不能铺线,"."表必须铺
\(\color{#0066ff}{输出格式}\)
输出一个整数,表示总方案数
\(\color{#0066ff}{输入样例}\)
4 4
**..
....
....
....
\(\color{#0066ff}{输出样例}\)
2
\(\color{#0066ff}{数据范围与提示}\)
none
\(\color{#0066ff}{ 题解 }\)
插头DP本来以为多niubility的算法原来本质还是个DP,就是情况多了点qwq
状压分割线,有三种情况,无,左插头,右插头(详见洛谷题解懒得写了)
只要明白状态分析出所有情况即可
#include<bits/stdc++.h>
#define LL long long
LL in() {
char ch; LL x = 0, f = 1;
while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
return x * f;
}
std::unordered_map<LL, LL> f[2];
//我TM就不写hash表
int n, m, s, t;
bool mp[100][100];
char getch() {
char ch = getchar();
while(ch != '*' && ch != '.') ch = getchar();
return ch;
}
void init() {
n = in(), m = in();
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++) {
mp[i][j] = getch() == '.';
if(mp[i][j]) s = i, t = j;
}
}
LL pos(int v, int x) { return (v << (x << 1)); }
//返回第x大块的v状态(两个二进制来状压)
LL work() {
int now = 0, nxt = 1;
f[0][0] = 1;
LL U = (1LL << ((m + 1) << 1)) - 1;
//全集
for(int i = 1; i <= n; i++) {
for(int j = 1; j <= m; j++) {
f[nxt].clear();
for(auto &k:f[now]) {
LL S = k.first, val = k.second;
LL L = (S >> ((j - 1) << 1)) & 3, R = (S >> (j << 1)) & 3;
//分割线(L是当前竖着的那个,R是紧接着横着的那个)
if(!mp[i][j]) {
if(!L && !R) f[nxt][S] += val;
continue;
}
// 0 0
if(!L && !R) {
if(mp[i][j + 1] && mp[i + 1][j]) f[nxt][S ^ pos(1, j - 1) ^ pos(2, j)] += val;
//0 0 -> 1 2
}
//2 1
else if(L == 2 && R == 1) {
//2 1 -> 0 0
f[nxt][S ^ pos(L, j - 1) ^ pos(R, j)] += val;
}
// 0 1 // 1 0 // 0 2 // 2 0
else if(!L || !R) {
//0 1 // 0 2
if(!L) {
//拐弯
if(mp[i][j + 1]) f[nxt][S] += val;
//不拐弯
if(mp[i + 1][j]) f[nxt][S ^ pos(L, j - 1) ^ pos(L, j) ^ pos(R, j - 1) ^ pos(R, j)] += val;
}
//同上
else if(!R) {
if(mp[i][j + 1]) f[nxt][S ^ pos(L, j - 1) ^ pos(L, j) ^ pos(R, j - 1) ^ pos(R, j)] += val;
if(mp[i + 1][j]) f[nxt][S] += val;
}
}
//1 1 // 2 2
else if(L == R) {
// 1 1
if(L == 1) {
int du = 0;
//1 1 -> 0 0 但是源头接口处右插头变成左插头
for(int p = j; ; p++) {
LL o = (S >> (p << 1)) & 3;
if(o == 1) du++;
if(o == 2) du--;
if(!du) {
//原来状态消去,弄上新状态
f[nxt][S ^ pos(L, j - 1) ^ pos(R, j) ^ pos(2, p) ^ pos(1, p)] += val;
break;
}
}
}
//根上面差不多
else if(L == 2) {
int du = 0;
for(int p = j - 1; ; p--) {
LL o = (S >> (p << 1)) & 3;
if(o == 1) du--;
if(o == 2) du++;
if(!du) {
f[nxt][S ^ pos(L, j - 1) ^ pos(R, j) ^ pos(2, p) ^ pos(1, p)] += val;
break;
}
}
}
}
//本状态当且仅当是终点,用于封口
else if(L == 1 && R == 2 && i == s && j == t) return val;
}
std::swap(now, nxt);
}
f[nxt].clear();
//末尾的竖分割线到了下一行就变成了行首的分割线,把状态《《给分割线腾出地方
for(auto &k:f[now]) f[nxt][(k.first << 2) & U] += k.second;
std::swap(now, nxt);
}
return 0;
}
int main() {
init();
printf("%lld\n", work());
return 0;
}
P5056 【模板】插头dp的更多相关文章
- 模板—插头dp(Ural 1519 Formula 1)
括号表示法: 据说比下一个要快而且灵活. #include<iostream> #include<cstring> #include<cstdio> #define ...
- 插头DP模板
/* 插头dp模板 抄的GNAQ 的 括号表示法 */ #include<cstdio> #include<algorithm> #include<cstring> ...
- 模板:插头dp
前言: 严格来讲有关dp的都不应该叫做模板,因为dp太活了,但是一是为了整理插头dp的知识,二是插头dp有良好的套路性,所以姑且还叫做模板吧. 这里先推荐一波CDQ的论文和这篇博客http://www ...
- bzoj1814 Ural 1519 Formula 1(插头dp模板题)
1814: Ural 1519 Formula 1 Time Limit: 1 Sec Memory Limit: 64 MBSubmit: 924 Solved: 351[Submit][Sta ...
- LG5056 【模板】插头dp
题意 题目背景 ural 1519 陈丹琦<基于连通性状态压缩的动态规划问题>中的例题 题目描述 给出n*m的方格,有些格子不能铺线,其它格子必须铺,形成一个闭合回路.问有多少种铺法? 输 ...
- 【模板】插头dp
题目描述 题解: 插头$dp$中经典的回路问题. 首先了解一下插头. 一个格子,上下左右四条边对应四个插头.就像这样: 四个插头. 一个完整的哈密顿回路,经过的格子一定用且仅用了两个插头. 所以所有被 ...
- [学习笔记]插头dp
基于连通性的状压dp 巧妙之处:插头已经可以表示内部所有状态了. 就是讨论麻烦一些. 简介 转移方法:逐格转移,分类讨论 记录状态方法:最小表示法(每次要重新编号,对于一类没用“回路路径”之类的题,可 ...
- 插头dp小结
插头dp: \(A:\)插头dp是什么? \(B:\)一种基于连通性状态压缩的动态规划问题 \(A:\)请问有什么应用呢? \(B:\)各种网格覆盖问题,范围允许状压解决,常用于计算方案数与联通块权值 ...
- 插头dp
插头dp 感受: 我觉得重点是理解,算法并不是直接想出怎样由一种方案变成另一种方案.而是方案本来就在那里,我们只是枚举状态统计了答案. 看看cdq的讲义什么的,一开始可能觉得状态很多,但其实灰常简单 ...
随机推荐
- MFC简单的橡皮筋程序
void CMainWindow::OnLButtonDown(UINT nFlags,CPoint point) { //以下三个是在CMainWindow中定义 m_ptFrom=point; m ...
- QT5提示can not find -lGL的解决方法
这是由于 Qt5.0 默认将OpenGL加入了工程,但是在机器上没有安装OpenGL,所以jonas只需要在机器上安装OpenGL即可 . 安装建立基本编译环境 首先不可或缺的,就是编译器与基本的 ...
- DAY9-python并发之多线程
一 threading模块介绍 multiprocess模块的完全模仿了threading模块的接口,二者在使用层面,有很大的相似性,因而不再详细介绍 官网链接:https://docs.python ...
- css之content
content 属性与 :before 及 :after 伪元素配合使用,来插入生成内容.该属性用于定义元素之前或之后放置的生成内容.默认地,这往往是行内内容,不过该内容创建的框类型可以用属性 dis ...
- springmvc 注解扫描失败的可能原因
情况是这样的:web工程采用了ssm框架,dao和service都是通过annotation方式注入的,工程运行正常.后来把service和dao打成jar放在工程的lib目录下,问题来了,配置没改动 ...
- 洛谷P3328(bzoj 4085)毒瘤线段树
题面及大致思路:https://www.cnblogs.com/Yangrui-Blog/p/9623294.html, https://www.cnblogs.com/New-Godess/p/45 ...
- 2018多校第九场1010 (HDU6424) 数学
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6424 解法:找规律.因为最多三项,a1^a2^a3可以拆成(a1+2)+(a2+1)*a3,然后建成数 ...
- opencv reshape函数说明
转自http://blog.csdn.net/yang6464158/article/details/20129991 reshape有两个参数: 其中,参数:cn为新的通道数,如果cn = 0,表示 ...
- 第四章输入/输出(I/O)4.1I/O涉及的设备及相关概念简介
PCL中所有的处理都是基于点云展开的,利用不同的设备获取点云.存储点云等都是点云处理前后必须做的流程,PCL中有自己设计的内部PCD文件格式,为此,设计读写该该格式以及与其他3D文件格式之间进行转化的 ...
- 形式化验证工具(PAT)Perterson Algorithm学习
今天学习一下Perterson Algorithm. 这个算法是使用三个变量来实现并发程序的互斥性算法. 具体看一下代码: Peterson算法是一个实现互斥锁的并发程序设计算法,核心就是三个标志位是 ...