题面:

传送门

思路:

插头dp基础教程

先理解一下题意:实际上就是要你求这个棋盘中的哈密顿回路个数,障碍不能走

看到这个数据范围,还有回路处理,就想到使用插头dp来做了

观察一下发现,这道题因为都是回路,所以联通块上方的插头一定两两配对,可以使用括号序列代替最小表示法

分情况讨论一下

情况一:当前格子上方和左方都没有插头

这种时候可以继续,也可以给当前格子加一个下插头一个右插头,相当于一个新的联通分量

情况二:上方有一个下插头,左边没有

这时有两个决策:可以向右转,也可以继续向下,操作就是分别给这个格子一个右插头或者一个下插头

注意此时新插头的括号类型和原来的那个插头相同(画个图可以理解一下)

情况三:左边有一个右插头,上面没有

同情况二,转弯或者直走

情况四:都有插头,而且两个插头是同一括号

这种情况,我们可以将这两个插头合并,在当前格子把这条路径封闭了

但是这里需要考虑一下其他的插头

我们去掉了两个相同的括号,就需要把另外一个括号反过来配对才行

比如当前的括号序列是 ((##()#())##),加粗的是我们要合并的两个括号,那么这两个)变成#以后,它们原来匹配的左括号(就失配了,需要其中一个(右边的那个)左括号变成右括号,两个重新配对

也就是((##()#())##)变成((##()#(####)变成((##()#)####)

当然也可以画个图理解一下,两条路径相当于是绕了圈接起来了

这个操作需要扫一遍整个序列,是$O\left(n\right)$的,当然也可以预处理变成$O\left(1\right)$

情况五:都有插头,且两个是)(

这时候直接合并就好了,图片同上(理解一下,博主懒得再画一个图了.......)

情况六:都有插头,而且两个是()

这种时候只有在最后一个非障碍格子才能合并,标志着路径完全封闭,得到了一个答案

图中的蓝色和绿色代表样例中的两条路径,再最后一个格子合并

状态数略多,可以滚动数组+哈希处理

分类讨论的时候注意可不可以这么做(需要判断下一个格子是否为障碍)

Code:

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
#define hash ddf
using namespace std;
int n,m,x[][],cur,pre,ex,ey;
int st[][];ll ans[][],re;
int tot[],bit[],state[],st_tot,hash=;
struct edge{
int to,next;
}a[];
void insert(int sta,ll val){
// cout<<"insert "<<sta<<ends<<val<<endl;
int p=sta%hash,i;
for(i=state[p];i;i=a[i].next){
if(st[cur][a[i].to]==sta){
ans[cur][a[i].to]+=val;return;
}
}
tot[cur]++;
a[++st_tot].to=tot[cur];
a[st_tot].next=state[p];
state[p]=st_tot;st[cur][tot[cur]]=sta;ans[cur][tot[cur]]=val;
}
int main(){
int i,j,k,l,now,down,right;ll val;char s[];
scanf("%d%d",&n,&m);
for(i=;i<=n;i++){
scanf("%s",s);
for(j=;j<m;j++)
if(s[j]=='.')
x[i][j+]=,ex=i,ey=j+;
}
for(i=;i<;i++) bit[i]=i<<;
cur=;tot[cur]=;ans[cur][]=;st[cur][]=;
for(i=;i<=n;i++){
for(j=;j<=tot[cur];j++) st[cur][j]<<=;
for(j=;j<=m;j++){
// cout<<"begin "<<i<<ends<<j<<endl;
st_tot=;memset(state,,sizeof(state));
pre=cur;cur^=;tot[cur]=;
for(k=;k<=tot[pre];k++){
now=st[pre][k];val=ans[pre][k];
down=(now>>bit[j-])%;right=(now>>bit[j])%;
// cout<<" from "<<now<<ends<<val<<ends<<down<<ends<<right<<endl;
if(!x[i][j]){
if(!down&&!right){
insert(now,val);continue;
}
}
else if(!down&&!right){
if(x[i][j+]&&x[i+][j])
insert(now+(<<bit[j-])+((<<bit[j])<<),val);
}
else if(!down&&right){
if(x[i][j+]) insert(now,val);
if(x[i+][j])
insert(now-right*(<<bit[j])+right*(<<bit[j-]),val);
}
else if(down&&!right){
if(x[i+][j]) insert(now,val);
if(x[i][j+])
insert(now+down*(<<bit[j])-down*(<<bit[j-]),val);
}
else if(down==&&right==){
int cnt=;
for(l=j+;l<=m;l++){
if((now>>bit[l])%==) cnt++;
if((now>>bit[l])%==) cnt--;
if(!cnt){
insert(now-(<<bit[l])-(<<bit[j])-(<<bit[j-]),val);
break;
}
}
}
else if(down==&&right==){
int cnt=;
for(l=j-;l>=;l--){
if((now>>bit[l])%==) cnt++;
if((now>>bit[l])%==) cnt--;
if(!cnt){
insert(now+(<<bit[l])-((<<bit[j])<<)-((<<bit[j-])<<),val);
break;
}
}
}
else if(down==&&right==){
insert(now-((<<bit[j-])<<)-(<<bit[j]),val);
}
else if(down==&&right==){
if(i==ex&&j==ey) re+=val;
}
}
}
}
printf("%lld\n",re);
}

[URAL1519] Formula 1 [插头dp入门]的更多相关文章

  1. URAL1519 Formula 1 —— 插头DP

    题目链接:https://vjudge.net/problem/URAL-1519 1519. Formula 1 Time limit: 1.0 secondMemory limit: 64 MB ...

  2. URAL Formula 1 ——插头DP

    [题目分析] 一直听说这是插头DP入门题目. 难到爆炸. 写了2h,各种大常数,ural垫底. [代码] #include <cstdio> #include <cstring> ...

  3. 【BZOJ1814】Ural 1519 Formula 1 插头DP

    [BZOJ1814]Ural 1519 Formula 1 题意:一个 m * n 的棋盘,有的格子存在障碍,求经过所有非障碍格子的哈密顿回路个数.(n,m<=12) 题解:插头DP板子题,刷板 ...

  4. 【Ural】1519. Formula 1 插头DP

    [题目]1519. Formula 1 [题意]给定n*m个方格图,有一些障碍格,求非障碍格的哈密顿回路数量.n,m<=12. [算法]插头DP [题解]<基于连通性状态压缩的动态规划问题 ...

  5. bzoj1814 Ural 1519 Formula 1(插头dp模板题)

    1814: Ural 1519 Formula 1 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 924  Solved: 351[Submit][Sta ...

  6. hdu 1693 插头dp入门

    hdu1693 Eat the Trees 题意 在\(n*m\)的矩阵中,有些格子有树,没有树的格子不能到达,找一条或多条回路,吃完所有的树,求有多少种方法. 解法 这是一道插头dp的入门题,只需要 ...

  7. 动态规划之插头DP入门

    基于联通性的状态压缩动态规划是一类非常典型的状态压缩动态规划问题,由于其压缩的本质并不像是普通的状态压缩动态规划那样用0或者1来表示未使用.使用两种状态,而是使用数字来表示类似插头的状态,因此.它又被 ...

  8. bzoj 1814 Ural 1519 Formula 1 ——插头DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1814 普通的插头 DP .但是调了很久.注意如果合并两个 1 的话,不是 “把向右第一个 2 ...

  9. Ural 1519 Formula 1 插头DP

    这是一道经典的插头DP单回路模板题. 用最小表示法来记录连通性,由于二进制的速度,考虑使用8进制. 1.当同时存在左.上插头的时候,需要判断两插头所在连通块是否相同,若相同,只能在最后一个非障碍点相连 ...

随机推荐

  1. Github的基本功能

    著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处.作者:Fadeoc Khaos链接:http://www.zhihu.com/question/20070065/answer/30 ...

  2. 如何着手学习一个新的PHP框架

    如今的PHP框架层出不穷,名气也各不相同.如何快速掌握一种框架?看看本文吧~ 如今的PHP框架层出不穷,名气也各不相同.我不是这方面的专家,甚至不能熟练地使用其中的一种,所以就不作推荐了.这里我要讨论 ...

  3. C++ 容器与继承

    如果容器类型定义为基类类型,那么虽然可以把派生类装进容器中,但是不能通过容器访问派生类自己的public成员,派生类将会倍切掉,只保留派生类的基类部分: 如果把容器定义为派生类类型,那么不能把基类类型 ...

  4. git出现误修改如何撤销

    场景1:当你改乱了工作区某个文件的内容,想直接丢弃工作区的修改时,用命令git checkout -- file. 场景2:当你不但改乱了工作区某个文件的内容,还添加到了暂存区时,想丢弃修改,分两步, ...

  5. 【原创】数据处理中判断空值的方法(np.isnan、is np.nan和pd.isna)比较

      转载请注明出处:https://www.cnblogs.com/oceanicstar/p/10869725.html  1.np.isnan(只有数组数值运算时可使用) 注意:numpy模块的i ...

  6. DevOps - 日志分析 -ELK

    wget --no-check-certificate --no-cookies --header "Cookie: oraclelicense=accept-securebackup-co ...

  7. LeetCode#453 最小移动次数使数组元素相等

    给定一个长度为 n 的非空整数数组,找到让数组所有元素相等的最小移动次数.每次移动可以使 n - 1 个元素增加 1. 示例: 输入: [,,] 输出: 解释: 只需要3次移动(注意每次移动会增加两个 ...

  8. ob缓存的基本使用

    在页面 加载的时候 如果 图片 很多 很大 会造成页面的阻塞降低用户体验 我们在点击页面的时候可以使用OB缓存 整个页面, 当用户点击的时候直接请求的是我们预先准备好的html页面 .也降低了我们数据 ...

  9. linux命令学习-2

    1. 对于已经在前台执行的命令,可以重新放到后台执行,首先按ctrl+z暂停已经运行的进程,然后使用jobs查看进程编号n:2. bg命令将停止的作业放到后台运行 bg %n3. kill -9 XX ...

  10. git之简单入门及操作~

    看了bili的教程,https://www.bilibili.com/video/av23853294?from=search&seid=3300012850779227291 特此整理下. ...