传送门

emm在雅礼集训的时候听到的一道题 上来就觉得是插头dp 最后果然是轮廓线状压233

我们简化一下题意。 有一个n*m的网格,每个格子是空地或障碍物,询问把每一个空地看成障碍物的情况下,用1*2的骨牌覆盖(可以留有空地)的方案数 对1e9+7取模 bzoj和洛咕题面都挂了233

我们发现留有空地就很烦,所以我们可以把空地看成1*1的骨牌,这样的话我们统计的方案数就是用1*1的骨牌和1*2的骨牌完全覆盖网格的方案数。

骨牌覆盖! ——》轮廓线状压!

但是我们发现如果对于每个格子直接计算的话 时间复杂度是O(n^3*2^m) 根本无法承受

所以我们考虑另一种做法 我们可以选择对前后缀进行合并这样的话复杂度就降到了O(n^2*2^m)

我们考虑如何对前后缀进行合并 即什么样的两条轮廓线是合法的


对应红色的格子作为我们的合并的格子的话 首先要求它上下两个格子已经被覆盖过了 然后就是上下对应的蓝绿格子应该状态相同 这样才能竖着填满棋盘(我们现在只考虑竖着因为横向的覆盖是在轮廓线dp的时候已经讨论过了)

所以我们对前后分别进行一次轮廓线dp(讨论横着放1*2竖着放1*2放1*1和不放) 然后最后统计答案的时候进行合并即可

附代码。(哦对bzoj卡空间只能开到1<<17不过也够了233)

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define inf 20021225
#define ll long long
#define mdn 1000000007
using namespace std; int f[18][18][1<<17],g[18][18][1<<17];
int bit[18],n,m,top;
int mp[18][18]; void add(int &x,int y){x=(x+y)%mdn;} void work()
{
//int top=(1<<m)-1;
f[1][1][top]=1;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
int x=i,y=j+1;
if(j==m) x=i+1,y=1;
if(x>n) continue;
for(int st=0;st<=top;st++)
if(f[i][j][st])
{
int tmp=f[i][j][st];
if((st&bit[j])==0&&mp[i][j]) continue;
if((st&bit[j])==0)
{
add(f[x][y][st|bit[j]],tmp);
continue;
}
if(mp[i][j]) add(f[x][y][st],tmp);
else
{
add(f[x][y][st],tmp);
add(f[x][y][st^bit[j]],tmp);
if(j>1&&(st&bit[j-1])==0) add(f[x][y][st|bit[j-1]],tmp);
}
}
} g[n][m][top]=1;
for(int i=n;i;i--)
for(int j=m;j;j--)
{
int x=i,y=j-1;
if(j==1) x=i-1,y=m;
if(x<1) continue;
for(int st=0;st<=top;st++)
if(g[i][j][st])
{
int tmp=g[i][j][st];
if((st&bit[j])==0&&mp[i][j]) continue;
//printf("%d %d %d %d\n",i,j,st,g[i][j][st]);
if((st&bit[j])==0)
{
add(g[x][y][st|bit[j]],tmp);
continue;
}
if(mp[i][j]) add(g[x][y][st],tmp);
else
{
add(g[x][y][st],tmp);
add(g[x][y][st^bit[j]],tmp);
if(j<m&&(st&bit[j+1])==0) add(g[x][y][st|bit[j+1]],tmp);
}
}
}
}
int main()
{
scanf("%d%d",&n,&m);
bit[1]=1;top=(1<<m)-1;
for(int i=2;i<=m;i++) bit[i]=bit[i-1]<<1;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",&mp[i][j]);
work();
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
if(mp[i][j])
{
printf("0 ");
continue;
}
int ans=0;
for(int s=0;s<=top;s++)
{
if(s&bit[j])
add(ans,(ll)f[i][j][s]*g[i][j][s]%mdn);
//printf("%d %d %d %d\n",i,j,f[i][j][s],g[i][j][s]);
}
printf("%d ",ans);
}
printf("\n");
} return 0;
}

BZOJ5205 [CodePlus 2018 3 月赛]白金元首与莫斯科的更多相关文章

  1. 【LibreOJ】#6354. 「CodePlus 2018 4 月赛」最短路 异或优化建图+Dijkstra

    [题目]#6354. 「CodePlus 2018 4 月赛」最短路 [题意]给定n个点,m条带权有向边,任意两个点i和j还可以花费(i xor j)*C到达(C是给定的常数),求A到B的最短距离.\ ...

  2. bzoj5204: [CodePlus 2018 3 月赛]投票统计(离散化+暴力)

    5204: [CodePlus 2018 3 月赛]投票统计 题目:传送门 题解: 谢谢niang老师的一道sui题 离散化之后直接搞啊(打完之后还错了...) 代码: #include<cst ...

  3. 【LibreOJ】#6299. 「CodePlus 2018 3 月赛」白金元首与克劳德斯

    [题意]给出坐标系中n个矩形,类型1的矩形每单位时间向x轴正方向移动1个单位,类型2的矩形向y轴正方向,初始矩形不重叠,一个点被矩形覆盖当且仅当它在矩形内部(不含边界),求$(-\infty ,+\i ...

  4. 「CodePlus 2018 3 月赛」白金元首与莫斯科

    $n \leq 17,m \leq 17$,$n*m$的01矩形,对每一个0问:当他单独变成1之后,在其他0处放多米诺牌(不一定放满,可以不放)的方案数.膜$1e9+7$. 直接$dp$是$n^42^ ...

  5. 「CodePlus 2018 3 月赛」白金元首与克劳德斯

    所有的云在此时没有重叠的面积 所有的云在此时没有重叠的面积 所有的云在此时没有重叠的面积 所有的云在此时没有重叠的面积 所有的云在此时没有重叠的面积 所有的云在此时没有重叠的面积 所有的云在此时没有重 ...

  6. LOJ#6354. 「CodePlus 2018 4 月赛」最短路[最短路优化建图]

    题意 一个 \(n\) 个点的完全图,两点之间的边权为 \((i\ xor\ j)*C\) ,同时有 \(m\) 条额外单向路径,问从 \(S\) 到 \(T\) 的最短路. \(n\leq 10^5 ...

  7. loj6300 「CodePlus 2018 3 月赛」博弈论与概率统计

    link 题意: A和B玩游戏,每轮A赢的概率为p.现在有T组询问,已知A赢了n轮输了m轮,没有平局,赢一局A得分+1,输一局得分-1,问A得分期望值? $n+m,T\leq 2.5\times 10 ...

  8. 【LibreOJ】#6298. 「CodePlus 2018 3 月赛」华尔兹 BFS

    [题意]给定n*m的网格,起点和终点位置,一些格指定下一步的方向,一些格任意.要求为方向任意的格确定方向,使起点可以走到终点.n,m<=50. [算法]BFS [题解]这道题最好用BFS,因为D ...

  9. bzoj 5283: [CodePlus 2018 3 月赛]博弈论与概率统计

    Description 大家的好朋友小 L 来到了博弈的世界.Alice 和 Bob 在玩一个双人游戏.每一轮中,Alice 有 p 的概率胜利,1 -p 的概率失败,不会出现平局.双方初始时各有 0 ...

随机推荐

  1. 前端-使用template-web.js【模版引擎】渲染数据

    通过ajax请求到的数据,使用传统的拼接字符串也可以做到,但是‘ “ ” ‘这种模式,单引号.双引号容易 扩错,新手入门推荐使用这种,初入前端,需要一个一个敲代码体会一下. 使用 template.j ...

  2. 项目中有 xxxx 不能被json序列化

    遇到这类问题 ,首先断点调试,看看要序列化的值 是一个什么类型的值 查看值得数据类型 在将值转化成可以被json序列化的对象 此时即可解决问题 如遇到  requests.post() 朝一个url发 ...

  3. 比传统事务快10倍?一张图读懂阿里云全局事务服务GTS

    近日,阿里云全局事务服务GTS正式上线,为微服务架构中的分布式事务提供一站式解决方案.GTS的原理是将分布式事务与具体业务分离,在平台层面开发通用的事务中间件GTS,由事务中间件协调各服务的调用一致性 ...

  4. PHP curl_file_create函数

    curl_file_create — 创建一个 CURLFile 对象. 说明 CURLFile curl_file_create ( string $filename [, string $mime ...

  5. git 小错误

    (一)在本地直接修改文件,提交后出现(master|REBASE 1/2).由于文件冲突所以导致各种报错. 在git pull --rebase origin master后 error: Pulli ...

  6. 测开之路五十七:实现runner和测试报告

    准备测试用例 from fox.case import Casefrom src import Calculator class TestCalculator(Case): def setUp(sel ...

  7. 取余运算 C和python的区别

    今天看书发现python与C的负数取余运算结果不同,查资料理解. 取余运算的算法是相同的  r = a- n*(a/n)   n!=0 r是余数,a是被除数,n是除数.n不能为0,否则都会报错. 负数 ...

  8. CSS属性去除图片链接时的虚线框

    CSS 之outline (轮廓)是绘制于元素周围的一条线,位于边框边缘的外围,可起到突出元素的作用.outline 属性是一个简写属性,用于设置元素周围的轮廓线.注释:轮廓线不会占据空间,也不一定是 ...

  9. Junit简单使用

    上篇文章也说到了,接口测试我使用的是Junit框架.因为我还是刚刚接触,也还没有深入了解,主要学会了断言的使用.断言是对测试运行结果的判断,如果结果符合要求,程序就会继续执行下去.反之,如果结果和预计 ...

  10. 将QTP运行时的错误截图上传到QC

    Class QCImageErrorCapture Sub Class_Terminate() 'Check if the current test has failed. If failed the ...