E. Arson In Berland Forest
The Berland Forest can be represented as an infinite cell plane. Every cell contains a tree. That is, contained before the recent events.

A destructive fire raged through the Forest, and several trees were damaged by it. Precisely speaking, you have a n×mn×m rectangle map which represents the damaged part of the Forest. The damaged trees were marked as "X" while the remaining ones were marked as ".". You are sure that all burnt trees are shown on the map. All the trees outside the map are undamaged.

The firemen quickly extinguished the fire, and now they are investigating the cause of it. The main version is that there was an arson: at some moment of time (let's consider it as 00) some trees were set on fire. At the beginning of minute 00, only the trees that were set on fire initially were burning. At the end of each minute, the fire spread from every burning tree to each of 88 neighboring trees. At the beginning of minute TT, the fire was extinguished.

The firemen want to find the arsonists as quickly as possible. The problem is, they know neither the value of TT (how long the fire has been raging) nor the coordinates of the trees that were initially set on fire. They want you to find the maximum value of TT (to know how far could the arsonists escape) and a possible set of trees that could be initially set on fire.

Note that you'd like to maximize value TT but the set of trees can be arbitrary.

Input

The first line contains two integer nn and mm (1≤n,m≤1061≤n,m≤106, 1≤n⋅m≤1061≤n⋅m≤106) — the sizes of the map.

Next nn lines contain the map. The ii-th line corresponds to the ii-th row of the map and contains mm-character string. The jj-th character of the ii-th string is "X" if the corresponding tree is burnt and "." otherwise.

It's guaranteed that the map contains at least one "X".

Output

In the first line print the single integer TT — the maximum time the Forest was on fire. In the next nn lines print the certificate: the map (n×mn×m rectangle) where the trees that were set on fire are marked as "X" and all other trees are marked as ".".

Examples
input

Copy
3 6
XXXXXX
XXXXXX
XXXXXX
output

Copy
1
......
.X.XX.
......
input

Copy
10 10
.XXXXXX...
.XXXXXX...
.XXXXXX...
.XXXXXX...
.XXXXXXXX.
...XXXXXX.
...XXXXXX.
...XXXXXX.
...XXXXXX.
..........
output

Copy
2
..........
..........
...XX.....
..........
..........
..........
.....XX...
..........
..........
..........
input

Copy
4 5
X....
..XXX
..XXX
..XXX
output

Copy
0
X....
..XXX
..XXX
..XXX

题意:给一个n*m<=1e6的图,"X"代表着火,"."代表没着火,火只在这个范围里面蔓延。着火的格子每回合会蔓延到周围的8个格子,给出了最后的火情图,求着火最多多少回合,并构造刚着火的图。

想法:很明显蔓延后的火都是一个矩形,所以我先横着遍历1遍,再竖着遍历遍,基本可以确认这个火最多蔓延了多少回合,构造图通过前缀和来判断,比较简单。然后我就被这组数据卡住了

7 5
..XXX
..XXX
..XXX
.XXX.
XXX..
XXX..
XXX.. 我一开始的输出
1
.....
...X.
.....
.....
.....
.X...
.....
答案
0
..XXX
..XXX
..XXX
.XXX.
XXX..
XXX..
XXX..

被这个数据卡住是因为,横着扫一遍竖着扫一遍只能确定最多蔓延回合不会超过这个数值,但是构造出来的图不一定能蔓延回原来的图。

做法:二分答案,先记录终图的前缀和,然后构造起始图,该点为"X"意味着终图中以该点为中心的m范围全为"X",用二维前缀和来判断。否则为"."。构造的同时记录起始图的前缀和。

构造完起始图后我们再检查初始是否合法,同样是用前缀和,该点为"X"意味着起始图中以该点为中心的m范围有"X"或者该点为"."意味着起始图中以该点为中心的m范围。

分析复杂度,每次二分我们需要构造1次和检查1次,复杂度是2e6。对于1e6的图需要二分20次复杂度是4e7,分析可得,蔓延最大不会超过1000回合,只需二分10次,复杂度变2e7。而横着扫一遍竖着扫一遍可以确定二分上界,这个操作复杂度是2e6,这样比纯二分快点。。

#include <bits/stdc++.h>
using namespace std;
const int N=2e6+;
char p[N];
char v[N];
int u[N];
int z[N];
int n,m;
inline bool solve(int zd)
{
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)
{
int t=i*(m+)+j;
v[t]='.';
if(p[t]=='X')
{
int flag=;
int rx=i+zd;
int ry=j+zd;
int lx=i-zd;
int ly=j-zd;
if(rx>n||ry>m||lx<=||ly<=)flag=;
if((u[rx*(m+)+ry]-u[rx*(m+)+ly-]-u[(lx-)*(m+)+ry])+u[(lx-)*(m+)+ly-]!=((zd*+)*(zd*+)))flag=;
if(!flag)v[t]='X';
}
z[t]=(v[t]=='X'?:)+z[t-]+z[t-m-]-z[t-m-];
}
}
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)
{
int t=i*(m+)+j;
if(p[t]=='X')
{
int rx=i+zd;
int ry=j+zd;
int lx=i-zd;
int ly=j-zd;
if(rx>n||ry>m||lx<=||ly<=)continue;
if((z[rx*(m+)+ry]-z[rx*(m+)+ly-]-z[(lx-)*(m+)+ry]+z[(lx-)*(m+)+ly-])==)return false;
}
}
}
return true;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)
{
int t=i*(m+)+j;
scanf(" %c",&p[t]);
u[t]=(p[t]=='X'?:)+u[t-]+u[t-m-]-u[t-m-];
}
}
int zd=;
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)
{
int t=;
while(p[i*(m+)+j]=='X')
{
j++;
t++;
if(j==m+)break;
}
if(t&&zd>t)zd=t;
}
}
for(int i=;i<=m;++i)
{
for(int j=;j<=n;++j)
{
int t=;
while(p[i+j*(m+)]=='X')
{
j++;
t++;
if(j==n+)break;
}
if(t&&zd>t)zd=t;
}
}
zd=(zd-)/;
if(zd==)
{
printf("0\n");
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)printf("%c",p[i*(m+)+j]);
printf("\n");
}
return ;
}
int l=,r=zd;
int ans=;
while(l<=r)
{
int m=(l+r)>>;
if(solve(m))
{
l=m+;
ans=m;
}
else r=m-;
}
printf("%d\n",ans);
solve(ans);
for(int i=;i<=n;++i)
{
for(int j=;j<=m;++j)printf("%c",v[i*(m+)+j]);
printf("\n");
}
return ;
}

代码

总结:直接开二维数组不可取,我是用了一维的来表示二维的,用vector应该会方便不少。然后就是要多练,此外对题目的复述要准确,问人时完全描述成了另一道题。。这都是值得努力的地方

Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) E. Arson In Berland Forest的更多相关文章

  1. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3

    A,有多个线段,求一条最短的线段长度,能过覆盖到所又线段,例如(2,4)和(5,6) 那么我们需要4 5连起来,长度为1,例如(2,10)(3,11),用(3,10) 思路:我们想一下如果题目说的是最 ...

  2. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) F2. Wrong Answer on test 233 (Hard Version) dp 数学

    F2. Wrong Answer on test 233 (Hard Version) Your program fails again. This time it gets "Wrong ...

  3. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) E. Arson In Berland Forest 二分 前缀和

    E. Arson In Berland Forest The Berland Forest can be represented as an infinite cell plane. Every ce ...

  4. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) D2. Optimal Subsequences (Hard Version) 数据结构 贪心

    D2. Optimal Subsequences (Hard Version) This is the harder version of the problem. In this version, ...

  5. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) C. Messy 构造

    C. Messy You are fed up with your messy room, so you decided to clean it up. Your room is a bracket ...

  6. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) B. Box 贪心

    B. Box Permutation p is a sequence of integers p=[p1,p2,-,pn], consisting of n distinct (unique) pos ...

  7. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) A. Math Problem 水题

    A. Math Problem Your math teacher gave you the following problem: There are n segments on the x-axis ...

  8. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) C Messy

    //因为可以反转n次 所以可以得到任何可以构成的序列 #include<iostream> #include<string> #include<vector> us ...

  9. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) B Box

    #include<bits/stdc++.h> using namespace std; ]; ]; int main() { int total; cin>>total; w ...

  10. Codeforces Round #602 (Div. 2, based on Technocup 2020 Elimination Round 3) A Math Problem

    //只要从所有区间右端点的最小值覆盖到所有区间左端点的最大值即可 #include<iostream> using namespace std ; int x,y; int n; int ...

随机推荐

  1. 吴裕雄--天生自然HADOOP学习笔记:hadoop集群实现PageRank算法实验报告

    实验课程名称:大数据处理技术 实验项目名称:hadoop集群实现PageRank算法 实验类型:综合性 实验日期:2018年 6 月4日-6月14日 学生姓名 吴裕雄 学号 15210120331 班 ...

  2. D. Cow and Snacks 并查集

    D. Cow and Snacks 题意:有n种小吃,m个人,每个人有两种喜欢的小吃,当一个人遇到两种自己都喜欢的小吃,可以都吃掉,问在最优的吃小吃顺序下,不能吃到自己喜欢的小吃的人数最少是多少? 题 ...

  3. mybatis注解中写SQL语句

    参考: https://blog.csdn.net/gebitan505/article/details/54929287/https://blog.csdn.net/KingBoyWorld/art ...

  4. ajax Ajax处理下载文件response没有反应

    参考:https://blog.csdn.net/wf632856695/article/details/52040034

  5. Python学习第一课——if-else

    #if 基本语句 if 1==1: print("如果条件为真,if执行该语句") else: print("如果条件为假,if则执行这条语句") #if 多重 ...

  6. 【Linux】centos7下解决yum -y install mysql-server 没有可用包

    第一步:安装从网上下载文件的wget命令 [root@localhost ~]# yum -y install wget 第二步:下载mysql的repo源 [root@localhost ~]# w ...

  7. 【Python数组及其基础操作】【numpy ndarray】

    一.创建数组 在python中创建数组最简单的办法就是使用array函数.它接受一切序列型的对象,然后产生一个含有传入数据的numpy数组.其中,嵌套序列(比如由一组等长列表组成的列表)会被转换为一个 ...

  8. 在linux中安装redis

    1.安装gcc环境: yum install gcc-c++ 2.下载redis的源码包,把源码包上传到linux服务器 3.解压源码包 :  tar -zxvf redis-3.0.0.tar.gz ...

  9. CH9 顺序容器

    本章主要介绍了标准库顺序容器,包括 顺序容器的公共接口,如构造函数,添加/删除操作等 利用迭代器访问容器 不同顺序容器的差异 string的特殊操作 容器适配器,如栈,队列等 9.1 “按字典序插入到 ...

  10. PHP 附录 : 用户注册与登录完整代码

    login.html <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http:// ...