1066: [SCOI2007]蜥蜴

Time Limit: 1 Sec Memory Limit: 162 MB

Submit: 2582 Solved: 1272

[Submit][Status][Discuss]

Description

在一个r行c列的网格地图中有一些高度不同的石柱,一些石柱上站着一些蜥蜴,你的任务是让尽量多的蜥蜴逃到边界外。 每行每列中相邻石柱的距离为1,蜥蜴的跳跃距离是d,即蜥蜴可以跳到平面距离不超过d的任何一个石柱上。石柱都不稳定,每次当蜥蜴跳跃时,所离开的石柱高度减1(如果仍然落在地图内部,则到达的石柱高度不变),如果该石柱原来高度为1,则蜥蜴离开后消失。以后其他蜥蜴不能落脚。任何时刻不能有两只蜥蜴在同一个石柱上。

Input

输入第一行为三个整数r,c,d,即地图的规模与最大跳跃距离。以下r行为石竹的初始状态,0表示没有石柱,1~3表示石柱的初始高度。以下r行为蜥蜴位置,“L”表示蜥蜴,“.”表示没有蜥蜴。

Output

输出仅一行,包含一个整数,即无法逃离的蜥蜴总数的最小值。

Sample Input

5 8 2

00000000

02000000

00321100

02000000

00000000

……..

……..

..LLLL..

……..

……..

Sample Output

1

HINT

100%的数据满足:1<=r, c<=20, 1<=d<=4

Source

Pku 2711 Leapin’ Lizards

这道题啊,我想啊。。(模仿自谢老板)
一眼就看出是网络流了,不过需要建图,一般网络流题目难点就在建图上,所以在建图上费了心
建立超级源,与各初始有蜥蜴的柱子连边,边权为1
建立超级汇,找可以跳出去的柱子与之连边,边权为INF
拆点:将每个柱子拆成两个点(入点出点)入点到出点边权为柱子高度
暴力将能够互相跳到的点连边(其中一点的出点连另一个点的入点),边权为INF
Dinic一遍得到最多能跳出的蜥蜴数,总数减掉即可。。。

妈妈我再也不会把数组开小了O(≧口≦)O

CODE:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int r,c,d;
int q[200000],h,t;
int dis[2000];
struct data{
int to,next,v;
}edge[500001];
int cnt=1,head[2000]={0};
int n;
int zh[25][25]={0};// 记录柱子高度
int xys[25][25]={0};//蜥蜴数
int bh[25][25]={0}; //编号(建图用)
int tot=0;//总蜥蜴数 void add(int u,int v,int w)
{
cnt++;
edge[cnt].next=head[u];
head[u]=cnt;
edge[cnt].to=v;
edge[cnt].v=w;
} void init()
{
scanf("%d%d%d",&r,&c,&d);
for (int i=1; i<=r; i++)
{
char x[30];
scanf("%s",&x);
for (int j=1; j<=c; j++)
zh[i][j]=x[j-1]-48;
}
for (int i=1; i<=r; i++)
{
char x[30];
scanf("%s",&x);
for (int j=1; j<=c; j++)
if (x[j-1]=='L')
{
tot++;
xys[i][j]=1;
}
}
}//读入 void make()//超级源点为1,超级汇为n(最后的num)
{
int num=2;
for (int i=1; i<=r; i++)
for (int j=1; j<=c; j++)
{
if (zh[i][j]>0)
{
bh[i][j]=num;//bh【】记录这个柱子的入点的编号(出点编号为入点编号+1)
add(num,num+1,zh[i][j]);
add(num+1,num,0);
num+=2;
}
}//拆点
for (int i=1; i<=r; i++)
for (int j=1; j<=c; j++)
{
if (xys[i][j]>0)
{
add(1,bh[i][j],1);
add(bh[i][j],1,0);
}
}//把初始有蜥蜴的与超级源连边
for (int i=1; i<=r; i++)
for (int j=1; j<=c; j++)
{
if (zh[i][j]>0)
{
for (int a=max(1,i-d); a<=min(r,i+d); a++)
for (int b=max(1,j-d); b<=min(c,j+d); b++)
{
if (zh[a][b]>0 && (a!=i || b!=j))
if ((a-i)*(a-i)+(b-j)*(b-j)<=d*d)
{
add(bh[i][j]+1,bh[a][b],0x7fffffff);
add(bh[a][b],bh[i][j]+1,0);
}
}
}
}//暴力把能调到的两个柱子连边
for (int i=1; i<=r; i++)
for (int j=1; j<=c; j++)
if (zh[i][j]>0 && (i-d<=0 || i+d>r || j-d<=0 || j+d>c))
{
add(bh[i][j]+1,num,0x7fffffff);
add(num,bh[i][j]+1,0);
}//把能跳出去的柱子与超级汇连边
n=num;
}//建图 bool bfs()
{
memset(dis,-1,sizeof(dis));
q[1]=1;dis[1]=1;
h=0; t=1;
while (h<t)
{
int j=q[++h],i=head[j];
while (i)
{
if (dis[edge[i].to]<0 && edge[i].v>0)
{
dis[edge[i].to]=dis[j]+1;
q[++t]=edge[i].to;
}
i=edge[i].next;
}
}
if (dis[n]>0)
return true;
else
return false;
} int dfs(int loc,int low)
{
int ans=0;
if (loc==n) return low;
int i=head[loc];
while (i)
{
if (edge[i].v>0 && dis[edge[i].to]==dis[loc]+1 && (ans=dfs(edge[i].to,min(low,edge[i].v))))
{
edge[i].v-=ans;
edge[i^1].v+=ans;
return ans;
}
i=edge[i].next;
}
return 0;
} int main()
{
init();
make();
int ans=0;
while (bfs())
{
int now;
while ((now=dfs(1,0x7fffffff)))
ans+=now;
}
printf("%d",tot-ans);
return 0;
}

BZOJ-1066 蜥蜴 最大流+拆点+超级源超级汇的更多相关文章

  1. BZOJ 1066 蜥蜴 最大流

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1066 题目大意: 在一个r行c列的网格地图中有一些高度不同的石柱,一些石柱上站着一些蜥 ...

  2. <每日一题>Day 9:POJ-3281.Dining(拆点 + 多源多汇+ 网络流 )

    Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 24945   Accepted: 10985 Descript ...

  3. bzoj 1066 蜥蜴

    最大流. 建图:首先将每根柱子拆成两个点. 每根柱子的入点向出点连一条容量为柱子高度的边. 每根柱子的出点向可以到达的柱子的入点连一条容量为正无穷的边. 源点向每根初始有蜥蜴的柱子的入点连一条容量为一 ...

  4. BZOJ 1066 蜥蜴(网络流)

    很普通的拆点网络流,把每个柱子拆成两个点(i,j,0)和(i,j,1).对于柱子的高度限制则加边((i,j,0),(i,j,1),height). 两个柱子能互相到达则加边((i,j,1),(i1,j ...

  5. poj--1459--Power Network(最大流,超级源超级汇)

    Power Network Time Limit: 2000MS   Memory Limit: 32768KB   64bit IO Format: %I64d & %I64u Submit ...

  6. [BZOJ 1066] [SCOI2007] 蜥蜴 【最大流】

    题目链接:BZOJ - 1066 题目分析 题目限制了高度为 x 的石柱最多可以有 x 只蜥蜴从上面跳起,那么就可以用网络流中的边的容量来限制.我们把每个石柱看作一个点,每个点拆成 i1, i2,从 ...

  7. POJ 2711 Leapin' Lizards / HDU 2732 Leapin' Lizards / BZOJ 1066 [SCOI2007]蜥蜴(网络流,最大流)

    POJ 2711 Leapin' Lizards / HDU 2732 Leapin' Lizards / BZOJ 1066 [SCOI2007]蜥蜴(网络流,最大流) Description Yo ...

  8. hdu2732 最大流+拆点

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2732 题目给定一个场景,有n*m个方格,每个方格代表一个柱子,一个柱子可以承受不同次数的跳跃,开始时图 ...

  9. BZOJ-1070 修车 最小费用最大流+拆点+略坑建图

    1070: [SCOI2007]修车 Time Limit: 1 Sec Memory Limit: 162 MB Submit: 3624 Solved: 1452 [Submit][Status] ...

随机推荐

  1. AC日记——石子归并 codevs 1048

    1048 石子归并  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解  查看运行结果     题目描述 Description 有n堆石子排成一列,每堆石子 ...

  2. 新手须知 C、C++和VC++之间的区别

    本文目地 本文介绍的是C.C++.VC++ 三者之间的区别,也许作为初学者来说,会很容易混淆.希望通过本文的介绍,能够给你带来帮助. C语言 C语言是一种古老而又经久不衰的计算机程序设计语言,大约诞生 ...

  3. [cb] Unity Editor 添加右键菜单

    需求 为Unity的Editor窗口添加右键菜单 实现代码 // This example shows how to create a context menu inside a custom Edi ...

  4. JMeter学习(二十五)HTTP属性管理器HTTP Cookie Manager、HTTP Request Defaults

    Test Plan的配置元件中有一些和HTTP属性相关的元件:HTTP Cache Manager.HTTP Authorization Manager.HTTP Cookie Manager.HTT ...

  5. 经典71道Android试题及答案

    本文为开发者奉献了70道经典Android面试题加答案--重要知识点几乎都涉及到了,你还等啥,赶紧收藏吧!! 1. 下列哪些语句关于内存回收的说明是正确的? (b) A. 程序员必须创建一个线程来释放 ...

  6. SQL语句统计每天、每月、每年的数据

    1.每年select year(ordertime) 年,sum(Total) 销售合计from 订单表group by year(ordertime) 2.每月select year(orderti ...

  7. where,having与 group by连用的区别

    select 列a,聚合函数 from 表名 where 过滤条件 group by 列a having 过滤条件 group by 字句也和where条件语句结合在一起使用.当结合在一起时,wher ...

  8. POJ 1125 Stockbroker Grapevine

    Stockbroker Grapevine Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 33141   Accepted: ...

  9. Linux Linux程序练习五

    题目:编写两个进程a和b,利用共享内存技术,a向共享内存写字符串,b将从共享内存中读到的字符串在屏幕上打印出来. //创建共享内存区 #include <stdio.h> #include ...

  10. CSS 实现加载动画之二-圆环旋转

    上次简单的介绍了下如何用代码实现菊花旋转的加载动画,动画点击,这次继续我们的动画系列,实现另外一种加载动画,圆环旋转.与上次不同的是,菊花旋转是通过改变元素透明度来实现动画,这次因为考虑到元素叠加,加 ...