这是一个非常神奇的题;

感觉像一个模拟搜索;

但是竟然可以用网络流来解决;

直接粘题解把:

  如果不能走通的话,必然说明能够从右上角(图外面)沿雷“跳” ,一直可以“跳”左
下角(图外面) ,因此建好图之后求一个最小割就可以得到结果了。但是关键在于:1.哪些
雷之间可以相互“跳” ?2.哪些雷可以从右上角“跳”过去,哪些雷可以“跳”到左下角?
  第二个问题很好办,如果地雷的范围能接触到最上或者最右的格子,就可以从右上角跳
到这个雷上,如果地雷的范围能接触到最下或者最右的格子,就可以跳到左下角。
  第一个问题需要分类讨论一下,如果两个雷在同一水平线或者竖直线上,当两个雷的距
离不超过 2*K+1 时可以认为两个雷是连通的,但如果不在同一水平线或者竖直线上时,当
两个雷的距离不超过 2*K+2 时可以认为两个雷是连通的。

注意雷要拆点,容量为 1,其他边容量为 INF

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define maxn 1550
#define maxm 40009
#define inf 9999999
using namespace std; struct edge
{
int from,to,cap,flow;
edge() {}
edge(int from,int to,int cap,int flow)
:from(from),to(to),cap(cap),flow(flow) {}
}; struct dinic
{
int n,m,s,t;
edge edges[maxm*];
int head[maxn];
int next[maxm*];
bool inq[maxn];
int d[maxn];
int cur[maxn];
void init(int n)
{
this->n=n;
m=;
memset(head,-,sizeof(head[])*(n+));
} void addedge(int from,int to,int cap)
{
next[m]=head[from];
edges[m]=edge(from,to,cap,);
head[from]=m++;
next[m]=head[to];
edges[m]=edge(to,from,,);
head[to]=m++;
}
bool bfs()
{
memset(inq,,sizeof(inq[])*(n+));
queue<int>q;
q.push(s);
d[s]=;
inq[s]=;
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=head[u]; i!=-; i=next[i])
{
edge& e=edges[i];
int v=e.to;
if(!inq[v]&&e.cap>e.flow)
{
inq[v]=;
d[v]=d[u]+;
q.push(v);
if(v==t)return ;
}
}
}
return ;
}
int dfs(int u,int a)
{
if(u==t||a==)return a;
int flow=,f;
for(int&i=cur[u]; i!=-; i=next[i])
{
edge& e=edges[i];
int v=e.to;
if(d[u]+==d[v]&&(f=dfs(v,min(a,e.cap-e.flow)))>)
{
e.flow+=f;
edges[i^].flow-=f;
flow+=f;
a-=f;
if(a==)break;
}
}
return flow;
}
int maxflow(int s,int t)
{
this->s=s;
this->t=t;
int flow=;
while(bfs())
{
memcpy(cur,head,sizeof(head[])*(n+));
flow+=dfs(s,inf);
}
return flow;
}
}; dinic solve; struct node
{
int x,y;
} no[maxn];
char s[];
int main()
{
// freopen("test0.in","r",stdin);
int t;
int n,m,d;
scanf("%d",&t);
while(t--)
{
int cnt=;
scanf("%d%d%d",&n,&m,&d);
for(int i=; i<=n; i++)
{
scanf("%s",s+);
for(int j=; j<=m; j++)
{
if(s[j]=='*')
{
no[cnt].x=i;
no[cnt++].y=j;
}
}
}
int sr=;
int tr=*cnt+;
solve.init(*cnt+);
for(int i=; i<cnt; i++)
{
solve.addedge(*i,*i+,);
if((no[i].x-)<=d||((m-no[i].y)<=d))
solve.addedge(sr,*i,inf-);
if((n-no[i].x)<=d||(no[i].y-)<=d)
solve.addedge(*i+,tr,inf-);
}
for(int i=; i<cnt; i++)
for(int j=; j<cnt; j++)
{
int dis=abs(no[i].x-no[j].x)+abs(no[i].y-no[j].y);
if(no[i].x==no[j].x||no[i].y==no[j].y)
{
if(dis<=*d+)
solve.addedge(*i+,*j,inf-);
}
else
{
if(dis<=*d+)
solve.addedge(*i+,*j,inf-);
}
}
printf("%d\n",solve.maxflow(sr,tr));
}
return ;
}

csuoj 1355: 地雷清除计划的更多相关文章

  1. CSU 1355 地雷清除计划

    题目链接:http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1355 好题,根本想不到是网络流. 模型如图: 假想从右上角到左下角有一条阻拦线,我们就是 ...

  2. Linux操作系统计划任务

    ++++++++++++++++++++++++++++++++++++++++++++++++标题:Linux操作系统的计划任务内容:计划任务分为单次任务和周期性任务,周期任务分为系统级计划任务和用 ...

  3. 三.cron计划任务

    • 用途:按照设置的时间间隔为用户反复执行某一项固 定的系统任务 • 软件包:cronie.crontabs • 系统服务:crond • 日志文件:/var/log/crond   • 使用 cro ...

  4. DBCC 命令2

    状态查询:收集和显示各类信息,状态检查. 如cachestats.pss.sqlmgrstats.memorystatus.proccache.freeproccache.freesystemcach ...

  5. 《Xenogears》(异度装甲)隐含的原型与密码

    <Xenogears>(异度装甲)隐含的原型与密码 X 彩虹按:一种高次元的“生命体”,因“事故”被抓来当成“超能源”,其实那不只是“无限的能源”而已,“它”是有意志的!在我们眼里看来,这 ...

  6. 【linux之crontab,启动】

    一.计划任务 atd at命令发布的任务计划 一次性的任务计划 at time ctrl+d 提交 time: 1.绝对时间:12:00 2.相对时间:+8 3.模糊时间:noon midnight ...

  7. DataGuard 单实例到RAC搭建

    背景简介: 本文为针对一次windows平台RAC数据库迁移至Linux平台RAC的笔记,基本步骤为: 1.搭建windows RAC到Linux 单实例数据库的DataGuard 2.做switch ...

  8. FtpCopy数据定时自动备份软件(FTP定时备份)

    1. 软件说明 FtpCopy是一款免费的FTP数据自动备份软件,如果FtpCopy对您有较大的帮助,欢迎捐赠我们,我们对您表示衷心的感谢! 如果有需求的话会一直更新下去,将软件做到极致! 有问题可直 ...

  9. 使用json改写网站

    效果图 json文件 https://raw.githubusercontent.com/sherryloslrs/timetable/master/timetable.json { "Ti ...

随机推荐

  1. 关于SWT常用组件(按钮,复选框,单选框(Button类))

    Button是SWT中最常用的组件.Button类的继承关系图: Button类的构造方法是newe Button(Composite parent,int style)它有两个参数: 第一个参数:是 ...

  2. 【转】SharePoint工作流中常用的方法

    SharePoint中使用工作流对象模型. SharePoint工作流对象模型在Microsoft.SharePoint.Workflow命名空间.你可以利用此对象模型对你的工作流进行编程.你可以启动 ...

  3. sql 理解视图

    可以看作是定义在sqlserver上的虚拟的表,本身并不存储数据,仅仅存储一个select语句和涉及的表的引用 通过视图,客户端不再需要知道底层表结构和其之间的关系,视图提供了一个统一访问数据的接口 ...

  4. Sqlserver 安装

    安装环境: SqlServer版本:Sql Server 2008 (安装包您应该已有准备) =============以下开始安装,多图,基本软件操作不做太多说明,注意查看图片=========== ...

  5. ASP.NET MVC模型绑定的6个建议(转载)

    ASP.NET MVC模型绑定的6个建议 发表于2011-08-03 10:25| 来源博客园| 31 条评论| 作者冠军 validationasp.netmvc.netasp 摘要:ASP.NET ...

  6. iOS开发,多个button数组,每个数组只能选中5项,多个数组只能选择3个。

    由于常用xib,所以不想用代码写那么多个button.而且也懒的算位置 直接xib拉线成四个数组.水果,零食,饮料,甜点. 入题实现的功能就是,在这四个数组之中只能在3个数组只选中5项.有点绕(就比如 ...

  7. oc 基础知识

    一.枚举 结构体 typedef enum{ sexMan, sexWoman }Sex;   tydedef struct{    int year;    int month;    int da ...

  8. Eclipse Memory Analysis进行堆转储文件分析

    生成堆转储文件 新建项目,设置Eclispe Java堆的大小: (1)限制Java堆大小:将最小值 -Xms参数与最大值-Xmx参数设置一样可避免堆的扩展         -Xmx20m -Xms2 ...

  9. RTTI(Runtime Type Information )

    RTTI 是“Runtime Type Information”的缩写,意思是:运行时类型信息.它提供了运行时确定对象类型的方法.本文将简略介绍 RTTI 的一些背景知识.描述 RTTI 的概念,并通 ...

  10. InstallShield 版本转换

    InstallShield : 如何用低版本 打开高版本的工程   InstallShield 每个版本都有对应的版本号SchemaVersion,如下所示   InstallShield Versi ...