URAL1519 Formula 1 【插头dp】
题目链接
题解
看题型显然插头\(dp\)
考虑如何设计状态
有这样一个方案
当我们决策到某个位置
轮廓线长这样
你会发现插头一定是相互匹配的
所以我们实际上可以把状态用括号序列表示
如上图就是(#)()
是一个三进制数
那么我们设\(f[i][j][s]\)表示决策到\((i,j)\),轮廓线状态为\(s\)的方案数
我们同时记\(0\)为空插头,\(1\)为表示左括号的插头,\(2\)为表示有括号的插头
先不管空间问题,我们考虑一下转移
有比较多的情况
我们记\(b1\),\(b2\)为\((i,j)\)的左、上插头
\(b1 = 0,b2 = 0\)
首先如果\((i,j)\)本身是障碍格,那么它右插头和下插头也为\(0\)
否则如果对应方向没有障碍,\((i,j)\)右下插头为\(12\)
\(b1\)和\(b2\)有一者为\(0\),那么转移的时候另一个括号的位置放哪里都可以,只需要判断有无障碍
\(b1\)和\(b2\)为同一种括号,我们只需往另一侧查找匹配的括号,改变方向
例如((#))变为###()
如图所示:
把左边两个连起来,右边两个插头就变成了匹配的括号
即由((#))变为###()
\(b1 = 2\)且\(b2 = 1\)
就是(#)(#)这种情况,可以变为(####)
\(b1 = 1\)且\(b2 = 2\)
除非是最后一个格子,否则不能贸然连起来,不然就会出现不连通的情况
具体实现的时候,可以使用四进制而结合位运算加快速度
由于空间比较小,我们需要滚动数组,并且使用\(hash\)表储存状态
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#include<vector>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 13,maxm = 5000000,INF = 1000000000,P = 201611;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int n,m,sx,sy;
char S[maxn][maxn];
int now,tot[2],h[2][P],nxt[2][maxm],num[2][maxm];
LL f[2][maxm],ans;
inline void add(int s,LL x){
int tmp = s % P;
for (int k = h[now][tmp]; k; k = nxt[now][k])
if (num[now][k] == s){f[now][k] += x; return;}
nxt[now][++tot[now]] = h[now][tmp]; h[now][tmp] = tot[now];
num[now][tot[now]] = s; f[now][tot[now]] = x;
}
inline bool isok(int x,int y){
return x >= 1 && x <= n && y >= 1 && y <= m && S[x][y] != '*';
}
void work(){
int las = 1,e,s,b1,b2;
LL x;
f[0][1] = tot[0] = 1; num[0][1] = 0;
for (int i = 1; i <= n; i++){
for (int k = 1; k <= tot[now]; k++) num[now][k] <<= 2;
for (int j = 1; j <= m; j++){
now ^= 1; las ^= 1;
cls(h[now]); tot[now] = 0;
for (int k = 1; k <= tot[las]; k++){
s = num[las][k]; x = f[las][k];
b1 = (s >> (j - 1 << 1)) & 3;
b2 = (s >> (j << 1)) & 3;
e = s ^ (b1 << (j - 1 << 1)) ^ (b2 << (j << 1));
if (b1 == 0 && b2 == 0){
if (S[i][j] == '*') add(e,x);
else if (isok(i + 1,j) && isok(i,j + 1))
add(e | (1 << (j - 1 << 1)) | (2 << (j << 1)),x);
}
else if (b1 == 0){
if (isok(i,j + 1)) add(s,x);
if (isok(i + 1,j)) add(e | (b2 << (j - 1 << 1)),x);
}
else if (b2 == 0){
if (isok(i + 1,j)) add(s,x);
if (isok(i,j + 1)) add(e | (b1 << (j << 1)),x);
}
else if (b1 == 1 && b2 == 1){
int cnt = 1;
for (int p = j + 1; p <= m + 1; p++){
if ((e >> (p << 1) & 3) == 1) cnt++;
if ((e >> (p << 1) & 3) == 2) cnt--;
if (!cnt){add(e ^ (3 << (p << 1)),x); break;}
}
}
else if (b1 == 2 && b2 == 2){
int cnt = 1;
for (int p = j - 2; ~p; p--){
if ((e >> (p << 1) & 3) == 2) cnt++;
if ((e >> (p << 1) & 3) == 1) cnt--;
if (!cnt){add(e ^ (3 << (p << 1)),x); break;}
}
}
else if (b1 == 2 && b2 == 1) add(e,x);
else if (i == sx && j == sy) ans += x;
}
}
}
printf("%lld\n",ans);
}
int main(){
n = read(); m = read();
REP(i,n) scanf("%s",S[i] + 1);
REP(i,n) REP(j,m) if (S[i][j] == '.') sx = i,sy = j;
work();
return 0;
}
URAL1519 Formula 1 【插头dp】的更多相关文章
- URAL1519 Formula 1 —— 插头DP
题目链接:https://vjudge.net/problem/URAL-1519 1519. Formula 1 Time limit: 1.0 secondMemory limit: 64 MB ...
- [URAL1519] Formula 1 [插头dp入门]
题面: 传送门 思路: 插头dp基础教程 先理解一下题意:实际上就是要你求这个棋盘中的哈密顿回路个数,障碍不能走 看到这个数据范围,还有回路处理,就想到使用插头dp来做了 观察一下发现,这道题因为都是 ...
- 【BZOJ1814】Ural 1519 Formula 1 插头DP
[BZOJ1814]Ural 1519 Formula 1 题意:一个 m * n 的棋盘,有的格子存在障碍,求经过所有非障碍格子的哈密顿回路个数.(n,m<=12) 题解:插头DP板子题,刷板 ...
- 【Ural】1519. Formula 1 插头DP
[题目]1519. Formula 1 [题意]给定n*m个方格图,有一些障碍格,求非障碍格的哈密顿回路数量.n,m<=12. [算法]插头DP [题解]<基于连通性状态压缩的动态规划问题 ...
- bzoj1814 Ural 1519 Formula 1(插头dp模板题)
1814: Ural 1519 Formula 1 Time Limit: 1 Sec Memory Limit: 64 MBSubmit: 924 Solved: 351[Submit][Sta ...
- bzoj 1814 Ural 1519 Formula 1 ——插头DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1814 普通的插头 DP .但是调了很久.注意如果合并两个 1 的话,不是 “把向右第一个 2 ...
- Ural 1519 Formula 1 插头DP
这是一道经典的插头DP单回路模板题. 用最小表示法来记录连通性,由于二进制的速度,考虑使用8进制. 1.当同时存在左.上插头的时候,需要判断两插头所在连通块是否相同,若相同,只能在最后一个非障碍点相连 ...
- URAL Formula 1 ——插头DP
[题目分析] 一直听说这是插头DP入门题目. 难到爆炸. 写了2h,各种大常数,ural垫底. [代码] #include <cstdio> #include <cstring> ...
- bzoj 1814 Ural 1519 Formula 1 插头DP
1814: Ural 1519 Formula 1 Time Limit: 1 Sec Memory Limit: 64 MBSubmit: 942 Solved: 356[Submit][Sta ...
- BZOJ1814: Ural 1519 Formula 1(插头Dp)
Description Regardless of the fact, that Vologda could not get rights to hold the Winter Olympic gam ...
随机推荐
- 【转】bash: ssh: command not found解决方法(linux)
原文转自:http://www.cnblogs.com/ahauzyy/archive/2013/04/25/3043699.html 今天在搭建hadoop的开发环境中,用的是centsos6.0的 ...
- Electron小记
一.安装 1.安装NodeJS 2.安装electronjs:npm install -g electron --unsafe-perm=true --allow-root 安装完,环境为: Node ...
- C#特性的简单介绍
特性应该我们大多接触过,比喻经常使用的[Obsolete],[Serializable]等下面我就主要介绍一个特性的一些用法 摘自MSDN定义:用以将元数据或声明信息与代码(程序集.类型.方法.属性等 ...
- textbox的验证
代码如下: textBox1.KeyDown += (a, b) => { if (b.KeyCode == Keys.Enter) { textBox2.Focus(); } }; textB ...
- MySQL数据库之安装,基本操作
一.基础部分 1.数据库是什么 之前所学,数据要永久保留,比如用户注册的用户信息,都是保存于文件,而文件只能存在于某一台机器上. 如果我们不考虑从文件中读取数据的效率问题,并且假设我们的程序所有的组件 ...
- VIN码识别/车架号识别独家支持云识别
VIN码(车架号)对于懂车的人来说并不陌生,不要小看这一串字符,从VIN码中可以读懂车辆的生产厂家.年代.车型.车身型式及代码.发动机代码及组装地点等信息. 一辆汽车的VIN码也是车辆的唯一身份证明, ...
- realstudio 粒子特效问题总结
ParticleEmitter._inner_material.flags.depth_write = true;ParticleEmitter._inner_material.flags.depth ...
- Blockchain For Dummies(IBM Limited Edition
Blockchain For Dummies(IBM Limited Edition)笔记 该系列内容主要介绍用于商业的区块链,有人说区块链之于贸易,犹如因特网之于信息.在商业领域区块链可以用于交易任 ...
- scrapy+selenium+chromedriver解析动态渲染页面
背景:动态页面是页面是通过js代码渲染出来的,无法直接使用scrapy爬虫,这是就需要先把js代码转为静态的html,再用scrapy爬虫就可以解决 解决办法:增加SeleniumMiddleware ...
- Scrum立会报告+燃尽图(十一月二十三日总第三十一次):界面修改及新页面添加
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2410 项目地址:https://git.coding.net/zhang ...