题目

给出 \(n*m\) 的方格,有些格子不能铺线,

其它格子必须铺,可以形成多个闭合回路。

问有多少种铺法? \(n,m\leq 12\)


分析

设 \(dp[n][m][S][0/1]\) 表示处理到 \((n,m)\),

目前插头的状态为 \(S\),并且左插头是否向右暴露,

插头的状态指的是轮廓线上的上插头是否向下暴露,这个分类讨论一下就可以了,

注意到右边界的位置左插头不能向右暴露,并且不能铺线的地方要继续更新方案


代码

#include <cstdio>
#include <cctype>
#include <cstring>
#define rr register
using namespace std;
int n,m,al,a[12][12];
long long dp[4101][2],f[4101][2];
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
signed main(){
for (rr int T=iut();T;--T){
memset(dp,0,sizeof(dp));
n=iut(),m=iut(),al=1<<m,dp[0][0]=1;
for (rr int i=0;i<n;++i)
for (rr int j=0;j<m;++j)
a[i][j]=iut();
for (rr int i=0;i<n;++i)
for (rr int j=0;j<m;++j){
if (a[i][j]){
for (rr int k=0;k<al;++k)
if ((k>>j)&1){
f[k][0]+=dp[k][0];//|
f[k^(1<<j)][1]+=dp[k][0];//L
f[k^(1<<j)][0]+=dp[k][1];//>
}else{
f[k][1]+=dp[k][1];//-
f[k|(1<<j)][0]+=dp[k][1];//7
f[k|(1<<j)][1]+=dp[k][0];//<
}
}else{
for (rr int k=0;k<al;++k)
if (!((k>>j)&1))
f[k][0]=dp[k][0];//不存在插头
}
if (j==m-1){
for (rr int k=0;k<al;++k)
f[k][1]=0;//不能有暴露的左插头
}
memcpy(dp,f,sizeof(dp));
memset(f,0,sizeof(f));
}
printf("%lld\n",dp[0][0]);
}
return 0;
}

简化

可以发现,dp数组能够被压成一维,记录 \(m+1\) 个插头,

轮廓线往右移动一格左插头也向右移动一个。

注意位于行末时需要移除右侧插头并补上左侧插头,即调整轮廓线的状态


代码(HDU 1693)

#include <cstdio>
#include <cctype>
using namespace std;
int T,two[14],n,m,al;
long long f[8192],dp[8192];
int iut(){
int ans=0; char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=ans*10+c-48,c=getchar();
return ans;
}
int main(){
T=iut(),two[0]=1;
for (int i=1;i<=13;++i) two[i]=two[i-1]<<1;
for (int _T=1;_T<=T;++_T){
n=iut(),m=iut(),al=two[m+1]-1;
for (int i=0;i<=al;++i) f[i]=0; f[0]=1;
for (int i=0;i<=al;++i) dp[i]=f[i];
for (int i=0;i<n;++i)
for (int j=0;j<m;++j){
int x=iut();
for (int S=0;S<=al;++S) f[S]=0;
for (int S=0;S<=al;++S)
if (dp[S]){
bool zuo=(S>>j)&1,upp=(S>>(j+1))&1;
if (x){
f[S^two[j]^two[j+1]]+=dp[S];//转弯
if (zuo!=upp) f[S]+=dp[S];//直行
}else if (!zuo&&!upp) f[S]+=dp[S];
}
for (int S=0;S<=al;++S)
if (j<m-1) dp[S]=f[S];
else dp[S]=(S&1)?0:f[S>>1];//行末最右侧的左插头不存在,插头整体往右移动一格,同时在左侧补上左插头
}
printf("Case %d: There are %lld ways to eat the trees.\n",_T,dp[0]);
}
return 0;
}

#插头dp#洛谷 5074 HDU 1693 Eat the Trees的更多相关文章

  1. hdu 1693 Eat the Trees——插头DP

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=1693 第一道插头 DP ! 直接用二进制数表示状态即可. #include<cstdio> # ...

  2. HDU 1693 Eat the Trees(插头DP、棋盘哈密顿回路数)+ URAL 1519 Formula 1(插头DP、棋盘哈密顿单回路数)

    插头DP基础题的样子...输入N,M<=11,以及N*M的01矩阵,0(1)表示有(无)障碍物.输出哈密顿回路(可以多回路)方案数... 看了个ppt,画了下图...感觉还是挺有效的... 参考 ...

  3. HDU 1693 Eat the Trees(插头DP)

    题目链接 USACO 第6章,第一题是一个插头DP,无奈啊.从头看起,看了好久的陈丹琦的论文,表示木看懂... 大体知道思路之后,还是无法实现代码.. 此题是插头DP最最简单的一个,在一个n*m的棋盘 ...

  4. HDU - 1693 Eat the Trees(多回路插头DP)

    题目大意:要求你将全部非障碍格子都走一遍,形成回路(能够多回路),问有多少种方法 解题思路: 參考基于连通性状态压缩的动态规划问题 - 陈丹琦 下面为代码 #include<cstdio> ...

  5. HDU 1693 Eat the Trees(插头DP,入门题)

    Problem Description Most of us know that in the game called DotA(Defense of the Ancient), Pudge is a ...

  6. HDU 1693 Eat the Trees (插头DP)

    题意:给一个n*m的矩阵,为1时代表空格子,为0时代表障碍格子,问如果不经过障碍格子,可以画一至多个圆的话,有多少种方案?(n<12,m<12) 思路: 这题不需要用到最小表示法以及括号表 ...

  7. HDU 1693 Eat the Trees ——插头DP

    [题目分析] 吃树. 直接插头DP,算是一道真正的入门题目. 0/1表示有没有插头 [代码] #include <cstdio> #include <cstring> #inc ...

  8. hdu 1693 : Eat the Trees 【插头dp 入门】

    题目链接 题意: 给出一个n*m大小的01矩阵,在其中画线连成封闭图形,其中对每一个值为1的方格,线要恰好穿入穿出共两次,对每一个值为0的方格,所画线不能经过. 参考资料: <基于连通性状态压缩 ...

  9. HDU 1693 Eat the Trees

    第一道(可能也是最后一道)插头dp.... 总算是领略了它的魅力... #include<iostream> #include<cstdio> #include<cstr ...

  10. 【HDU】1693 Eat the Trees

    http://acm.hdu.edu.cn/showproblem.php?pid=1693 题意:n×m的棋盘求简单回路(可以多条)覆盖整个棋盘的方案,障碍格不许摆放.(n,m<=11) #i ...

随机推荐

  1. PMP考试计算题汇总

    第6章 项目时间管理 本节术语较多.涉及的工具&技术也不少. 主要包括活动定义.活动排序.活动资源估算.活动历时估算.进度制定.进度控制6个子过程. 1.1活动定义:就是对WBS的进一步分解. ...

  2. macOS通过ssh使用PEM登录

    在win上面可以使用XSHELL来登录类似于亚马逊这样的安全服务器,在mac上面就可以使用系统自带的命令工具来连接 使用命令 ssh -i key.pem [server] 如下: ssh -i ke ...

  3. 50条MAUI踩坑记

    1. 目录结构: (1)_imports.razor是一个全局using namespace的地方 (2)Platforms下的代码,虽然都放在同一个项目下,但是Platforms\Android下的 ...

  4. Java 常用类 于 StringBuffer 和 StringBuilder的使用 + String三者的异同

    1 package com.bytezero.stringclass; 2 3 import org.junit.Test; 4 5 /** 6 * 关于 StringBuffer 和 StringB ...

  5. Java 自定义数组的工具类

    1 /** 2 * 3 * @Description 自定义数组的工具类 4 * @author Bytezero·zhenglei! Email:420498246@qq.com 5 * @vers ...

  6. tp5.1 controller 名称自动转换大小写,导致文件名对不上 url_convert

    // 是否自动转换URL中的控制器和操作名 'url_convert' => false, // true,

  7. vxe table columns 要用data里的值,用computed的值会导致排序部分不好用。

    vxe table columns 要用data里的值,用computed的值会导致排序部分不好用.

  8. 基于BES2300芯片的开源DSP开发平台简述

    一 什么是开源DSP平台 所谓的开源DSP平台,就是软硬件全部开发接口,开发者可以在上面做DSP算法验证和算法开发.基于目前科研机构缺少开源的微型数字信号处理的情况,我们把bes2300的代码做了优化 ...

  9. Spring Boot 2.0 新人会踩的坑--启动报404错误

    转载自:http://www.javaman.cn/channels/sb2 启动程序,验证效果 根据图示,点击按钮,来启动 Spring Boot Web 程序, 查看控制台输出: . ____ _ ...

  10. Shell 编写摘要 (一)

    PS:要转载请注明出处,本人版权所有. PS: 这个只是基于<我自己>的理解, 如果和你的原则及想法相冲突,请谅解,勿喷. 前置说明   本文作为本人csdn blog的主站的备份.(Bl ...