Educational Codeforces Round 27 F. Guards In The Storehouse
1.5 seconds
512 megabytes
standard input
standard output
Polycarp owns a shop in the capital of Berland. Recently the criminal activity in the capital increased, so Polycarp is thinking about establishing some better security in the storehouse of his shop.
The storehouse can be represented as a matrix with n rows and m columns. Each element of the matrix is either . (an empty space) or x (a wall).
Polycarp wants to hire some guards (possibly zero) to watch for the storehouse. Each guard will be in some cell of matrix and will protect every cell to the right of his own cell and every cell to the bottom of his own cell, until the nearest wall. More formally, if the guard is standing in the cell (x0, y0), then he protects cell (x1, y1) if all these conditions are met:
- (x1, y1) is an empty cell;
- either x0 = x1 and y0 ≤ y1, or x0 ≤ x1 and y0 = y1;
- there are no walls between cells (x0, y0) and (x1, y1). There can be a guard between these cells, guards can look through each other.
Guards can be placed only in empty cells (and can protect only empty cells). The plan of placing the guards is some set of cells where guards will be placed (of course, two plans are different if there exists at least one cell that is included in the first plan, but not included in the second plan, or vice versa). Polycarp calls a plan suitable if there is not more than one empty cell that is not protected.
Polycarp wants to know the number of suitable plans. Since it can be very large, you have to output it modulo 109 + 7.
The first line contains two numbers n and m — the length and the width of the storehouse (1 ≤ n, m ≤ 250, 1 ≤ nm ≤ 250).
Then n lines follow, ith line contains a string consisting of m characters — ith row of the matrix representing the storehouse. Each character is either . or x.
Output the number of suitable plans modulo 109 + 7.
1 3
.x.
3
2 2
xx
xx
1
2 2
..
..
10
3 1
x
.
x
2
In the first example you have to put at least one guard, so there are three possible arrangements: one guard in the cell (1, 1), one guard in the cell (1, 3), and two guards in both these cells.
题意:
在地图上放士兵守卫, " . " 代表那个位置可以放士兵 , " x " 代表那个位置是墙
士兵的守卫范围为:
① 士兵右侧连续的的 " . " 直到墙或者地图边缘
② 士兵下方连续的的 " . " 直到墙或者地图边缘
③ 士兵所处的位置
求 放置士兵使得士兵守卫的不到的位置 <=1 的方案数 (不包括墙)
思路:
由于 1 ≤ nm ≤ 250 所以始终 min ( n,m ) ≤ 15 这是个有用的信息
容易发现 , 放完左上方位置的士兵后左上的情况不会受下面或者右边的影响
因此可以考虑dp
dp [ i ] 表示 按如图编号顺序 从上到下 从左往右 放置第 i 个士兵的局面

状态有两种
1)当前局面已经有一个位置守卫不到
2)当前所有位置都能被守护
但是我们发现一个问题 我们 怎样知道当前位置 放置/不放置 士兵会造成什么影响呢
所以我们需要知道 两个东西
1)上方是否有士兵能守卫到当前位置
2)左侧是否有士兵能守卫到当前位置
这时候对第一点 考虑到状压dp ( 因为min(n,m)<=15 ) 2进制的当前位为 1 则上方有士能守卫到这个位置 为 0 则没有
对第二点考虑增加一维 表示 dp 的状态 1 代表左侧有士能守卫到这个位置 0 则没有
所以 dp方程变成了 4 维 dp [ i ] [ j ] [ k ] [ h ] 表示
放置第 i 个士兵 , j 代表2进制表示上方的状态 , k 代表 是否左侧有士兵能守卫当前位置 0为没有 1为有 , h表示已经被考虑过的位置有多少位置无法被守卫
转移方式如下图

紫色线的左侧和上方为被考虑过的位置 红色为当前位置 蓝色为放置完士兵
那么 图中状态就是
dp [ i ] [ j ] [ k ] [ h ]
i = 9
j = 1110 (二进制) ///分别表示 第一列 第二列 第三列 第四列
k = 0 ( 左侧无士兵 )
h = 0 (没有被守卫到的位置为 0 个)
如果当前位置放置士兵 转移到

dp [ i ] [ j ] [ k ] [ h ]
i = 10
j = 1110 (二进制) ///分别表示 第一列 第二列 第三列 第四列
k = 1 ( 左侧有士兵 )
h = 0 (没有被守卫到的位置为 0 个)
否则转移到

dp [ i ] [ j ] [ k ] [ h ]
i = 10
j = 1110 (二进制) ///分别表示 第一列 第二列 第三列 第四列
k = 0 ( 左侧无士兵 )
h = 0 (没有被守卫到的位置为 0 个)
注意到两点
① 放行末的士兵时 由于要转移到下一行的第一个 那么 他转移时所有状态中的( dp[ i ] [ j ] [ k ] [h ] ) k 只能是 0
② 由于 min (n , m ) <=15 所以当列数比较多时要把 地图行列翻转 使得列数<=15
具体的转移情况稍微看一下就出来了 代码中也有
为了节省空间 转移时可以用滚动数组把dp的第一维降下来
代码:
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <stdlib.h>
#include <assert.h>
#include <math.h>
#include <set>
#include <map>
#include <stack>
#include <queue>
#include <vector>
#include <string>
#include <iostream>
#include <algorithm>
#include <functional> #define mp make_pair
#define pb push_back
#define mes(a,b) memset(a,b,sizeof(a))
#define mes0(a) memset(a,0,sizeof(a))
#define lson l,mid,pos<<1
#define rson mid+1,r,pos<<1|1
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define fi first
#define se second
#define sss(a) a::iterator
#define all(a) a.begin(),a.end() using namespace std; typedef double DB;
typedef long long LL;
typedef pair<int,int> pii;
typedef pair<long long ,int> pli;
typedef pair<int,long long > pil;
typedef pair<string,int> psi;
typedef pair<long long ,long long > pll; const int inf = 0x3f3f3f3f;
const long long INF = 0x3f3f3f3f3f3f3f3f;
const double pi = acos(-1.0);
const int maxn = +;
const int mod = 1e9+;
char MAP[][];
int n,m;
int dp[][(<<)][][]; /// 滚动 , 状压 , 左侧状态, 已填完的是否有空
int main()
{
dp[][][][]=;
int op=; ///滚动
scanf("%d%d",&n,&m);
if (m<=n){
for (int i=;i<n;i++){
scanf("%s",MAP[i]);
}
}else {
for (int i=;i<n;i++){
getchar();
for (int j=;j<m;j++){
MAP[j][i]=getchar();
}
}
swap(n,m);
}
for (int i=;i<n;i++){
int j;
for (j=;j<m;j++){
int flag=(j!=m-);
int now=m-j-;
int k;
memset(dp[op^],,sizeof(dp[op^]));
if (MAP[i][j]=='.'){
for (k=;k<(<<m);k++){
///dp[op][k][w][h];
/// m 1/0 是否填进去
/// 滚动数组, 状压 , 左侧状态, 已填完的是否有空
for (int m=;m<;m++){
for (int w=;w<;w++){
int st=(k>>now)&;
int S=k|((m|st)<<now);
st=(m|w|st)^;
for (int h=;h<;h++){
dp[op^][S][(w|m)&flag][h+st]+=dp[op][k][w][h];
dp[op^][S][(w|m)&flag][h+st]%=mod;
}
}
}
}
op^=;
} else {
for (k=;k<(<<m);k++){
///dp[op][k][w][h];
/// m 1/0 是否填进去
/// 滚动数组 , 状压 , 左侧状态, 已填完的是否有空
for (int w=;w<;w++){
int st=(k>>now)&;
int S=(k^(st<<now));
for (int h=;h<;h++){
dp[op^][S][][h]+=dp[op][k][w][h];
dp[op^][S][][h]%=mod;
}
}
}
op^=;
}
}
}
int ans=;
for (int i=;i<(<<m);i++){
for (int j=;j<;j++){
ans+=dp[op][i][][j];
ans%=mod;
}
}
cout<<ans<<endl;
return ;
}
Educational Codeforces Round 27 F. Guards In The Storehouse的更多相关文章
- Educational Codeforces Round 40 F. Runner's Problem
Educational Codeforces Round 40 F. Runner's Problem 题意: 给一个$ 3 * m \(的矩阵,问从\)(2,1)$ 出发 走到 \((2,m)\) ...
- Educational Codeforces Round 27 补题
题目链接:http://codeforces.com/contest/845 A. Chess Tourney 水题,排序之后判断第n个元素和n+1个元素是不是想等就可以了. #include < ...
- Educational Codeforces Round 61 F 思维 + 区间dp
https://codeforces.com/contest/1132/problem/F 思维 + 区间dp 题意 给一个长度为n的字符串(<=500),每次选择消去字符,连续相同的字符可以同 ...
- Educational Codeforces Round 51 F. The Shortest Statement(lca+最短路)
https://codeforces.com/contest/1051/problem/F 题意 给一个带权联通无向图,n个点,m条边,q个询问,询问两点之间的最短路 其中 m-n<=20,1& ...
- Educational Codeforces Round 12 F. Four Divisors 求小于x的素数个数(待解决)
F. Four Divisors 题目连接: http://www.codeforces.com/contest/665/problem/F Description If an integer a i ...
- Educational Codeforces Round 26 F. Prefix Sums 二分,组合数
题目链接:http://codeforces.com/contest/837/problem/F 题意:如题QAQ 解法:参考题解博客:http://www.cnblogs.com/FxxL/p/72 ...
- Educational Codeforces Round 9 F. Magic Matrix 最小生成树
F. Magic Matrix 题目连接: http://www.codeforces.com/contest/632/problem/F Description You're given a mat ...
- Educational Codeforces Round 6 F. Xors on Segments 暴力
F. Xors on Segments 题目连接: http://www.codeforces.com/contest/620/problem/F Description You are given ...
- Educational Codeforces Round 7 F. The Sum of the k-th Powers 拉格朗日插值法
F. The Sum of the k-th Powers 题目连接: http://www.codeforces.com/contest/622/problem/F Description Ther ...
随机推荐
- [LOJ6281]数列分块入门 5
题目大意: 给你一个长度为$n(n\leq50000)$的序列$A(0\leq A_i<2^{31})$,支持进行以下两种操作: 1.将区间$[l,r]$中所有数开方: 2.询问区间$[l,r] ...
- C++中数据对齐问题。struct、union、enum,类继承。再谈sizeof()
首先是struct,在C++中,结构体其实和class有很大的相似了.但是有一点不同的是,struct默认是public,而class中是private. 当然,struct继承等用法也是可以的. 共 ...
- 关于spring.net的面向切面编程 (Aspect Oriented Programming with Spring.NET)-使用工厂创建代理(Using the ProxyFactoryObject to create AOP proxies)
本文翻译自Spring.NET官方文档Version 1.3.2. 受限于个人知识水平,有些地方翻译可能不准确,但是我还是希望我的这些微薄的努力能为他人提供帮助. 侵删. 如果你正在为你的业务模型使用 ...
- Microsoft Office Excel 2007 使用笔记
1.显示表格边框: 选择要显示边框的单元格,点击“开始”选项卡中的“边框”图标,选中下拉框中的“所有框线” 2.单元格内,文字自动换行: 点击“开始”选项卡中的“自动换行”按钮 3.单元格内,文字手动 ...
- DEV MarqueeProgressBarControl控件
原文地址:http://www.dobug.net/showtopic-672.html MarqueeProgressBarControl是DevExpress的一个进度条控件,该控件和Progre ...
- 【Mybatis】mybatis查询报错org.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'areaName' in 'class java.lang.String'
mybatis查询报错: Caused by: org.apache.ibatis.reflection.ReflectionException: There is no getter for pro ...
- 2016summer 训练第二场
1.http://acm.hdu.edu.cn/showproblem.php?pid=5112 #define _CRT_SECURE_NO_DEPRECATE #include<iostre ...
- log4j教程 5、示例程序
前面我们已经看到了如何创建一个配置文件.本教程将讲解如何生成调试信息和日志在一个简单的文本文件. 下面是我们的例子中创建了一个简单的配置文件.这里再重复一次: 下载最新的Log4j库:http://l ...
- CSDN日报20170413 ——《天天写业务代码的那些年,我们是怎样成长过来的》
[程序人生]天天写业务代码的那些年,我们是怎样成长过来的 作者:Phodal 比起写业务代码更不幸的是,主要工作是修 Bug , bug , buG , bUg. [Java 编程]Springboo ...
- 前端存储之indexedDB
在前一个阶段的工作中,项目组要开发一个平台,为了做出更好的用户体验,实现快速.高质量的交互,从而更快得到用户的反馈,要求在前端把数据存储起来,之后我去研究了下现在比较流行的前端存储数据库,找到了ind ...