pid=3046">点击打开链接

题目:在一个N * M 的矩阵草原上,分布着羊和狼。每一个格子仅仅能存在0或1仅仅动物。如今要用栅栏将全部的狼和羊分开。问怎么放,栅栏数放的最少,求出个数?

解析:将狼群看作一个集合,羊群看作一个集合。然后设置源点和汇点,将两点至存在动物的点的距离赋值为1,构图,因为求得是栅栏数,从存在动物的位置向四周发散点赋值为1,即该方向放置一个栅栏。然后能够发现变成了求最小割,即求出最大流。须要注意的是,因为数据比較大,200 * 200。假设设置源点和汇点相差较大(即s = 0,e = n * m ),easyTLE.

#include<iostream>
#include<cstdio>
#include<cstring> using namespace std; // 最大流 ISAP + bfs+栈优化 const int maxn = 100010; //点的个数
const int maxm = 400010; //边的个数
const int INF = 0xfffffff; struct Edge{
int to, next, cap, flow;
}edge[ maxm ];//注意是maxm int tol;
int head[ maxn ];
int gap[ maxn ], dep[ maxn ], cur[ maxn ]; void init(){
tol = 0;
memset( head, -1, sizeof( head ) );
} void addedge( int u, int v, int w, int rw = 0 ){
edge[ tol ].to = v; edge[ tol ].cap = w; edge[ tol ].flow = 0;
edge[ tol ].next = head[ u ]; head[ u ] = tol++;
edge[ tol ].to = u; edge[ tol ].cap = rw; edge[ tol ].flow = 0;
edge[ tol ].next = head[ v ]; head[ v ] = tol++;
} int Q[ maxn ]; void BFS( int start, int end ){
memset( dep, -1, sizeof( dep ) );
memset( gap, 0, sizeof( gap ) );
gap[ 0 ] = 1;
int front = 0, rear = 0;
dep[ end ] = 0;
Q[ rear++ ] = end;
while( front != rear ){
int u = Q[ front++ ];
for( int i = head[ u ]; i != -1; i = edge[ i ].next ){
int v = edge[ i ].to;
if( dep[ v ] != -1 )
continue;
Q[ rear++ ] = v;
dep[ v ] = dep[ u ] + 1;
gap[ dep[ v ] ]++;
}
}
} int S[ maxn ];
int sap( int start, int end, int N ){
BFS( start, end );
memcpy( cur, head, sizeof( head ) );
int top = 0;
int u = start;
int ans = 0;
while( dep[ start ] < N ){
if( u == end ){
int Min = INF;
int inser;
for( int i = 0; i < top; ++i ){
if( Min > edge[ S[ i ] ].cap - edge[ S[ i ] ].flow ){
Min = edge[ S[ i ] ].cap - edge[ S[ i ] ].flow;
inser = i;
}
}
for( int i = 0; i < top; ++i ){
edge[ S[ i ] ].flow += Min;
edge[ S[ i ] ^ 1 ].flow -= Min;
}
ans += Min;
top = inser;
u = edge[ S[ top ] ^ 1 ].to;
continue;
}
bool flag = false;
int v;
for( int i = cur[ u ]; i != -1; i = edge[ i ].next ){
v = edge[ i ].to;
if( edge[ i ].cap - edge[ i ].flow && dep[ v ] + 1 == dep[ u ] ){
flag = true;
cur[ u ] = i;
break;
}
}
if( flag ){
S[ top++ ] = cur[ u ];
u = v;
continue;
}
int Min = N;
for( int i = head[ u ]; i != -1; i = edge[ i ].next ){
if( edge[ i ].cap - edge[ i ].flow && dep[ edge[ i ].to ] < Min ){
Min = dep[ edge[ i ].to ];
cur[ u ] = i;
}
}
gap[ dep[ u ] ]--;
if( !gap[ dep[ u ] ] )
return ans;
dep[ u ] = Min + 1;
gap[ dep[ u ] ]++;
if( u != start )
u = edge[ S[ --top ] ^ 1 ].to;
}
return ans;
} int dir[ ][ 2 ] = { { -1, 0 }, { 1, 0 }, { 0, -1 }, { 0, 1 } }; int main(){
int N, M, Case = 1;
while( scanf( "%d%d", &N, &M ) != EOF ){
int s = N * M, t = N * M + 1;
int n = N * M + 2, temp;
init();
for( int i = 0; i < N; ++i ){
for( int j = 0; j < M; ++j ){
scanf( "%d", &temp );
if( temp == 2 ){
addedge( s, M * i + j, INF );
}
if( temp == 1 ){
addedge( M * i + j, t, INF );
}
for( int k = 0; k < 4; ++k ){
int x = i + dir[ k ][ 0 ];
int y = j + dir[ k ][ 1 ];
if( x >= 0 && x < N && y >= 0 && y < M )
addedge( M * i + j, M * x + y, 1 );
}
}
}
printf( "Case %d:\n%d\n", Case++, sap( s, t, n ) );
}
return 0;
}

Pleasant sheep and big big wolf的更多相关文章

  1. HDU 3046 Pleasant sheep and big big wolf(最小割)

    HDU 3046 Pleasant sheep and big big wolf 题目链接 题意:一个n * m平面上,1是羊.2是狼,问最少要多少围墙才干把狼所有围住,每有到达羊的路径 思路:有羊和 ...

  2. Pleasant sheep and big big wolf HDU - 3046(最小割)

    Pleasant sheep and big big wolf Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 ...

  3. HDU 3046 Pleasant sheep and big big wolf

    Pleasant sheep and big big wolf Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged ...

  4. hdu 3046 Pleasant sheep and big big wolf 最小割

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3046 In ZJNU, there is a well-known prairie. And it a ...

  5. HDU 3046 Pleasant sheep and big wolf(最小割最大流+Dinic)

    http://acm.hdu.edu.cn/showproblem.php?pid=3046 题意: 给出矩阵地图和羊和狼的位置,求至少需要建多少栅栏,使得狼不能到达羊. 思路:狼和羊不能到达,最小割 ...

  6. hdu-3046-Pleasant sheep and big big wolf(最大流最小割)

    题意: 给出最少栏杆数使狼和羊分离 分析: 将狼与源点连,羊与汇点连,容量都为无穷,将图的各个相邻点连接,容量为1 然后题目就转化成最小去掉多少条边使不连通,即求最小割最大流. // File Nam ...

  7. HDU 3046Pleasant sheep and big big wolf(切最小网络流)

    职务地址:HDU 3046 最小割第一发!事实上也没什么发不发的. ..最小割==最大流.. 入门题,可是第一次入手最小割连入门题都全然没思路... sad..对最小割的本质还是了解的不太清楚.. 这 ...

  8. HDU3046_Pleasant sheep and big big wolf

    给一个n*m的数字阵,1表示羊的位置,2表示狼的位置,0表示没有东西,可以通过.在每个格子的4边都可以建立围栏,有围栏的话狼是不能通过的. 现在求最少建立多少围栏能够保证狼无法接触到羊. 题目的模型很 ...

  9. 【转载】图论 500题——主要为hdu/poj/zoj

    转自——http://blog.csdn.net/qwe20060514/article/details/8112550 =============================以下是最小生成树+并 ...

随机推荐

  1. 【Codeforces Round #476 (Div. 2) [Thanks, Telegram!] A】Paper Airplanes

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 统计每个人需要的sheet个数. 乘上k 然后除p就是需要的pack个数了 [代码] #include <bits/stdc+ ...

  2. 笔试中java的输入输出

    一,输入 import java.util.*; import java.io.*; public class Main { public static void main(String[] args ...

  3. HDU 4313 Contest 2

    很明显的树形DP了.但网上有的说可以用并查集.... 考虑一棵子树,当根结点有机器人时,则必定所有子树都要和根结点断开,而根结点向上返回的路径值则为其父结点与根结点连边的权值. 当根结点安全时,假设其 ...

  4. 用​M​y​E​c​l​i​p​s​e​ ​打​包​J​A​R文件

    用​M​y​E​c​l​i​p​s​e​ ​将自己定义标签打​成​J​A​R​包 1.新建一个javaproject 2.将标签有关的java代码拷贝到新建javaproject的一个包中,这时会报错 ...

  5. Ubuntu17.04安装WineQQ7.8及微信

    安装qq2012成功,但是提示版本过低,qq登录失败. 安装WineQQ WineQQ7.8下载 安装依赖软件,方法来源网上 32位ubuntu:sudo apt install libgtk-3-0 ...

  6. 图片3d轮放查看效果

    本功能比較简单,就是一个大幕.左右滚动播放图片. 关键点在于怎样实现平滑的滚动,包含动画效果,3d效果等. <style> img { position: absolute; top:20 ...

  7. 反射 + 配置文件 实现IOC容器

    IOC实现: IOC容器我们只停留在知道上是不行的,我们要动手做印象对更深刻,那么我给大家看一个代码.看看代码中IOC容器的实现. 代码实现: 创建一个类库: 解决方式的类库建立: 创建一个实体类:U ...

  8. node18---Mongoose

    二.索引index 数据库中,根据一个字段的值,来寻找一个文档,是很常见的操作.比如根据学号来找一个学生. 这个学号,是唯一的,只要有学号,就能唯一确认一个学生的文档.学号这个属性,就非常适合建立索引 ...

  9. poj_3071概率dp

    确定好对手就简单了. #include<iostream> #include<cstdio> #include<cstring> #include<algor ...

  10. 89.[NodeJS] Express 模板传值对象app.locals、res.locals

    转自:https://blog.csdn.net/Elliott_Yoho/article/details/53537437 locals是Express应用中 Application(app)对象和 ...