【BZOJ1814】Ural 1519 Formula 1 (插头dp)

题面

BZOJ

Vjudge

题解

戳这里

上面那个链接里面写的非常好啦。

然后说几个点吧。

首先是关于为什么只需要考虑三进制状态,因为哈密顿回路是不可能出现自交的,因此对于当前的轮廓线一定直接分割了哈密顿回路的一部分,不可能出现只考虑分割出来的情况下,存在插头的连通性直接交叉,否则一定不合法(比如说四个连续位置,你不可能\(1,3\)匹配,因为这样子画一条路径出来,无论如何都会和\(2,4\)的路径相交)。因此,我们把一组匹配看成左右括号,再加上有的位置可能不存在插头,所以是左括号、右括号、无插头三种情况。

然后是关于转移中的一部分转移,当左侧和上方的边都有插头连进来的时候,此时当前格子内的插头就固定了,同时转移的时候就得到了两个无插头的位置。注意一下,如果当前位置分别是\(1,1\)或者\(2,2\)的时候,你还需要修改其他位置的插头使他们满足左右匹配的关系,举个例子,本来是\(1122\)的匹配关系,你现在把\(22\)给连在一起了,于是这个状态就变成了\(1100\),所以你需要把第二个\(1\)变成\(2\),也就是这个状态应该是\(1200\)。

我的程序直接刚的三进制的,自己本机测没有问题,但是常数比较大,交到\(bzoj\)上就直接\(TLE\)了。可能直接改成四进制用位运算反而会快些。懒得改了,就直接贴代码吧。

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstdlib>
  4. #include<cstring>
  5. #include<cmath>
  6. #include<algorithm>
  7. #include<vector>
  8. using namespace std;
  9. #define ll long long
  10. #define MAX 15
  11. inline int read()
  12. {
  13. int x=0;bool t=false;char ch=getchar();
  14. while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
  15. if(ch=='-')t=true,ch=getchar();
  16. while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
  17. return t?-x:x;
  18. }
  19. int n,m,ln,lm,g[MAX][MAX];
  20. char ch[MAX];
  21. ll f[13][13][42000];
  22. int bin[MAX];
  23. int zt[42000],q[1600000],tot;
  24. ll ans;
  25. void dfs(int x,int c,int S)
  26. {
  27. if(c<0||c>m-x+1)return;
  28. if(x>m){zt[++tot]=S;q[S]=tot;return;}
  29. dfs(x+1,c,S);
  30. dfs(x+1,c+1,S+bin[x]);
  31. dfs(x+1,c-1,S+bin[x]+bin[x]);
  32. }
  33. int nxt(int x,int p)
  34. {
  35. for(int i=p,c=0;i<=m;++i)
  36. {
  37. if(x/bin[i]%3==2)--c;
  38. if(x/bin[i]%3==1)++c;
  39. if(!c)return i;
  40. }
  41. return m+1;
  42. }
  43. int pre(int x,int p)
  44. {
  45. for(int i=p,c=0;~i;--i)
  46. {
  47. if(x/bin[i]%3==1)--c;
  48. if(x/bin[i]%3==2)++c;
  49. if(!c)return i;
  50. }
  51. return m+1;
  52. }
  53. int main()
  54. {
  55. n=read();m=read();
  56. for(int i=1;i<=n;++i)
  57. {
  58. scanf("%s",ch+1);
  59. for(int j=1;j<=m;++j)
  60. if(ch[j]=='.')g[i][j]=1,ln=i,lm=j;
  61. }
  62. bin[0]=1;for(int i=1;i<=m;++i)bin[i]=bin[i-1]*3;
  63. dfs(0,0,0);f[1][0][q[0]]=1;
  64. for(int i=1;i<=n;++i)
  65. {
  66. for(int j=1;j<=m;++j)
  67. for(int k=1;k<=tot;++k)
  68. {
  69. int left=zt[k]/bin[j-1]%3,up=zt[k]/bin[j]%3;
  70. if(!g[i][j]){if(!left&&!up)f[i][j][k]+=f[i][j-1][k];}
  71. else
  72. {
  73. if(!left&&!up&&g[i][j+1]&&g[i+1][j])
  74. f[i][j][q[zt[k]+bin[j-1]+2*bin[j]]]+=f[i][j-1][k];
  75. if(!left&&up)
  76. {
  77. if(g[i][j+1])f[i][j][k]+=f[i][j-1][k];
  78. if(g[i+1][j])f[i][j][q[zt[k]+up*(bin[j-1]-bin[j])]]+=f[i][j-1][k];
  79. }
  80. if(left&&!up)
  81. {
  82. if(g[i+1][j])f[i][j][k]+=f[i][j-1][k];
  83. if(g[i][j+1])f[i][j][q[zt[k]+left*(bin[j]-bin[j-1])]]+=f[i][j-1][k];
  84. }
  85. if(left==1&&up==1)f[i][j][q[zt[k]-bin[j-1]-bin[j]-bin[nxt(zt[k],j)]]]+=f[i][j-1][k];
  86. if(left==2&&up==1)f[i][j][q[zt[k]-2*bin[j-1]-bin[j]]]+=f[i][j-1][k];
  87. if(left==2&&up==2)f[i][j][q[zt[k]-2*bin[j-1]-2*bin[j]+bin[pre(zt[k],j-1)]]]+=f[i][j-1][k];
  88. if(left==1&&up==2&&i==ln&&j==lm)ans+=f[i][j-1][k];
  89. }
  90. }
  91. for(int j=1;j<=tot;++j)
  92. if(zt[j]%3==0)f[i+1][0][j]+=f[i][m][q[zt[j]/3]];
  93. }
  94. printf("%lld\n",ans);
  95. return 0;
  96. }

【BZOJ1814】Ural 1519 Formula 1 (插头dp)的更多相关文章

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

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

  2. BZOJ1814: Ural 1519 Formula 1(插头Dp)

    Description Regardless of the fact, that Vologda could not get rights to hold the Winter Olympic gam ...

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

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

  4. 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 ...

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

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

  6. Ural 1519 Formula 1 插头DP

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

  7. bzoj 1814: Ural 1519 Formula 1 插头dp经典题

    用的括号序列,听说比较快. 然并不会预处理,只会每回暴力找匹配的括号. #include<iostream> #include<cstdio> #include<cstr ...

  8. bzoj1814 Ural 1519 Formula 1(插头DP)

    对插头DP的理解还不是很透彻. 先说一下肤浅的理解吧. 插头DP使用范围:指数级复杂度,且适用于解决网格图连通性问题,如哈密顿回路等问题.插头一般指每相邻2个网格的接口. 题目难度:一般不可做. 使用 ...

  9. 插头DP讲解+[BZOJ1814]:Ural 1519 Formula 1(插头DP)

    1.什么是插头$DP$? 插头$DP$是$CDQ$大佬在$2008$年的论文中提出的,是基于状压$D$P的一种更高级的$DP$多用于处理联通问题(路径问题,简单回路问题,多回路问题,广义回路问题,生成 ...

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

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

随机推荐

  1. 把List<T>转换为DataTable

    下面这个学习,把List<T>转换为Datatable. 下面先创建一个对象T: class Ay { private int _ID; public int ID { get { ret ...

  2. 如何使用串口来给STM32下载程序

    前言 第一次学习STM32的时候,不知道有调试器这个东西,所以一直是通过串口来给STM32下载程序,下载速度也还算可以,一般是几秒钟完成.后来用了调试器,可以直接在Keil环境下进行下载,而且还可以进 ...

  3. Linux系统特殊变量

    系统给定的特殊变量: 变量名 作用 $0 当前脚本的名字 $n 传递给脚本或者函数的参数,n表示第几个参数 $# 传递给脚本或函数的参数个数 $* 传递给脚本或函数的所有参数 $@ 传递给脚本或者函数 ...

  4. 浅谈java反射机制

    目录 什么是反射 初探 初始化 类 构造函数 属性 方法 总结 思考 什么是反射 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意 ...

  5. c#基础系列2---深入理解 String

    "大菜":源于自己刚踏入猿途混沌时起,自我感觉不是一般的菜,因而得名"大菜",于自身共勉. 扩展阅读:深入理解值类型和引用类型 基本概念 string(严格来说 ...

  6. python2.6升级到3.3.0 以及依赖库在迁移时的处理

    线上服务器python版本默认是2.6,由于业务程序要求,需要将python升级到3.3.0, 操作记录如下: Cenots6.8默认安装的是2.6版本,要更新升级需安装下gcc: [root@ope ...

  7. LINUX内核分析第八周学习总结——进程的切换和系统的一般执行过程

    LINUX内核分析第八周学习总结——进程的切换和系统的一般执行过程 张忻(原创作品转载请注明出处) <Linux内核分析>MOOC课程http://mooc.study.163.com/c ...

  8. 结对项目https://github.com/bxoing1994/test/blob/master/源代码

    所选项目名称:文本替换      结对人:曲承玉 github地址 :https://github.com/bxoing1994/test/blob/master/源代码 结对人github地址:ht ...

  9. PAT 1014 福尔摩斯的约会

    https://pintia.cn/problem-sets/994805260223102976/problems/994805308755394560 大侦探福尔摩斯接到一张奇怪的字条:“我们约会 ...

  10. HTML 5 placeHolder

    <html> <body> <input type="text" id="idNum" placeholder="pla ...