狼抓兔子 bzoj-1001 BeiJing2006

Description

现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,
而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形:

左上角点为(1,1),右下角点为(N,M)(上图中N=4,M=5).有以下三种类型的道路 
1:(x,y)<==>(x+1,y) 
2:(x,y)<==>(x,y+1) 
3:(x,y)<==>(x+1,y+1) 
道路上的权值表示这条路上最多能够通过的兔子数,道路是无向的. 左上角和右下角为兔子的两个窝,
开始时所有的兔子都聚集在左上角(1,1)的窝里,现在它们要跑到右下解(N,M)的窝中去,狼王开始伏击
这些兔子.当然为了保险起见,如果一条道路上最多通过的兔子数为K,狼王需要安排同样数量的K只狼,
才能完全封锁这条道路,你需要帮助狼王安排一个伏击方案,使得在将兔子一网打尽的前提下,参与的
狼的数量要最小。因为狼还要去找喜羊羊麻烦.

Input

第一行为N,M.表示网格的大小,N,M均小于等于1000.
接下来分三部分
第一部分共N行,每行M-1个数,表示横向道路的权值. 
第二部分共N-1行,每行M个数,表示纵向道路的权值. 
第三部分共N-1行,每行M-1个数,表示斜向道路的权值. 
输入文件保证不超过10M

Output

输出一个整数,表示参与伏击的狼的最小数量.

Sample Input

3 4
5 6 4
4 3 1
7 5 3
5 6 7 8
8 7 6 5
5 5 5
6 6 6

Sample Output

14

想法:显然,这题想求最小割。直接求最小割非常慢,我们考虑将它转换成对偶图。所谓对偶图,就是对于一个边和边之间没有交点的平面图来讲,对于每一个被边围出来的平面都看做一个点,如果原来平面上的边分割了两个平面,那么就将它对应的两个平面所构成的点之间连一条这条边权的边即可。然后跑最短路。

特别地,我们将start和end之间额外连一条边。然后我们将这条新连的边的将原来平面分成的两部分设为最短路中的七点和中点即可。

最后,附上丑陋的代码... ...在网上抄的代码(不会写对偶图)

#include <iostream>
#include <cstdio>
#include <queue>
using namespace std;
const int MAXV=2e6+105,MAXE=3e6+5,inf=0x7f7f7f7f;
int N,M,S,T;
struct E
{
int next,to,val;
}e[MAXE<<1];
int ecnt,G[MAXV];
void Edge(int u,int v,int w)
{
e[++ecnt]=(E){G[u],v,w};G[u]=ecnt;
e[++ecnt]=(E){G[v],u,w};G[v]=ecnt;
}
struct Node
{
int id,v;
bool operator<(const Node & ot)const
{return v>ot.v;}
};
priority_queue<Node> q;
bool inS[MAXV];int dis[MAXV];
int dijkstra()
{
for(int i=1;i<=T;i++)
{
dis[i]=inf;
}
dis[S]=0;
q.push((Node){S,0});
while(!q.empty())
{
int u=q.top().id;
q.pop();
if(inS[u]) continue;
inS[u]=true;
for(int i=G[u];i;i=e[i].next)
{
int v=e[i].to;
if(inS[v]) continue;
if(dis[v]>dis[u]+e[i].val)
{
dis[v]=dis[u]+e[i].val;
q.push((Node){v,dis[v]});
}
}
}
return dis[T];
}
int main()
{
int w;
scanf("%d%d",&N,&M);
if(N==1||M==1)
{
if(N>M) swap(N,M);
int ans=inf;
for(int i=1;i<=M;i++)
scanf("%d",&w),ans=min(ans,w);
printf("%d\n",ans);
}
else
{
S=2*(N-1)*(M-1)+1,T=S+1;
for(int i=1;i<M;i++)
{
int v=i*2;
scanf("%d",&w);
Edge(S,v,w);
}
for(int i=2;i<N;i++)
{
for(int j=1;j<M;j++)
{
int u=2*((i-2)*(M-1)+j)-1,v=2*((i-1)*(M-1)+j);
scanf("%d",&w);
Edge(u,v,w);
}
}
for(int i=1;i<M;i++)
{
int u=2*((N-2)*(M-1)+i)-1;
scanf("%d",&w);
Edge(u,T,w);
}
for(int i=1;i<N;i++)
{
for(int j=1;j<=M;j++)
{
scanf("%d",&w);
if(j==1)
{
int u=2*((i-1)*(M-1)+j)-1;
Edge(u,T,w);
}
else if(j==M)
{
int v=2*((i-1)*(M-1)+j-1);
Edge(S,v,w);
}
else
{
int u=2*((i-1)*(M-1)+j-1),v=u+1;
Edge(u,v,w);
}
}
}
for(int i=1;i<N;i++)
{
for(int j=1;j<M;j++)
{
int u=2*((i-1)*(M-1)+j)-1,v=u+1;
scanf("%d",&w);
Edge(u,v,w);
}
}
printf("%d\n",dijkstra());
}
return 0;
}

[bzoj1001][BeiJing2006]狼抓兔子_网络流_最小割转对偶图的更多相关文章

  1. BZOJ1001 BeiJing2006 狼抓兔子 【网络流-最小割】*

    BZOJ1001 BeiJing2006 狼抓兔子 Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,而且现在的兔子还比较 ...

  2. 【BZOJ1001】狼抓兔子(网络流)

    [BZOJ1001]狼抓兔子(网络流) 题面 Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的, 而且现在的兔子还比较笨, ...

  3. bzoj1001: [BeiJing2006]狼抓兔子(初识是你最小割)

    1001: [BeiJing2006]狼抓兔子 题目:传送门 题解: 听说这题当初是大难题...可惜当年没有网络流hahahha 现在用网络流的思想就很容易解决了嘛 给什么连什么,注意是双向边,然后跑 ...

  4. BZOJ1001: [BeiJing2006]狼抓兔子(优化的dinic或转化对偶图求最短路)

    1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 30078  Solved: 7908[Submit][ ...

  5. [BZOJ1001][BeiJing2006]狼抓兔子(最小割转最短路|平面图转对偶图)

    1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 31805  Solved: 8494[Submit][ ...

  6. BZOJ1001: [BeiJing2006]狼抓兔子 [最小割 | 对偶图+spfa]

    1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 19528  Solved: 4818[Submit][ ...

  7. BZOJ1001: [BeiJing2006]狼抓兔子【最短路+对偶图】

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1001 1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Se ...

  8. bzoj1001: [BeiJing2006]狼抓兔子 -- 最小割

    1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MB Description 现在小朋友们最喜欢的"喜羊羊与灰太狼 ...

  9. Bzoj1001 [BeiJing2006]狼抓兔子

    Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 19759  Solved: 4883 Description 现在小朋友们最喜欢的"喜羊羊与 ...

随机推荐

  1. 【POJ 1011】 Sticks

    [题目链接] http://poj.org/problem?id=1011 [算法] 深搜剪枝 首先我们枚举木棍的长度i,那么就有s/i根木棍,其中s为木棍长度的总和,朴素的做法就是对每种长度进行搜索 ...

  2. [HDU4689]Derangement

    https://zybuluo.com/ysner/note/1232641 题面 给出\(b_1,b_2,...,b_n\in\{−1,1\}\),求满足\((p_i−i)*b_i<0\)的\ ...

  3. Hamming Distance(随机算法)

    http://acm.hdu.edu.cn/showproblem.php?pid=4712 题意:计算任意两个十六进制的数异或后1的最少个数. 思路:用随机数随机产生两个数作为下标,记录这两个数异或 ...

  4. 七牛php上传下载类,集成官方文档的方法

    <?phpuse Qiniu\Auth;use Qiniu\Storage\UploadManager;class qiniu{ public $_accesskey = null; publi ...

  5. 洛谷P2756 飞行员配对方案问题(二分图匹配)

    P2756 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其 ...

  6. C/C++中输入多组数据方法

    --------开始-------- 对于刚开始学编程的人来说每次基本上就是一次数据输入,多次的话基本也是会给定一个数组的大小,但随着做刷算法题开始,题目有的会不直接告诉输入几组数据,基本输入都是多组 ...

  7. JS中对象按属性排序(冒泡排序)

    在实际工作经常会出现这样一个问题:后台返回一个数组中有i个json数据,需要我们根据json中某一项进行数组的排序. 例如返回的数据结构大概是这样: { result:[ {id:,name:'中国银 ...

  8. HTML多媒体标记之字幕标记

    在HTML中,可以向页面中插入字幕,水平或垂直滚动显示文字信息,字幕标记的格式如下: <marquee 属性="值"...>滚动的文字信息</marquee> ...

  9. git的常用命令。。

    git的常用命令.. git help <command>  显示command的help git show  显示某次提交的内容 git show $id git co -- <f ...

  10. git上

    ## 建立本地版本库 ## 本地版本库与远程关联 ## 修改文件并提交 ## 创建分支,修改文件合并至master 1. git的由来 linux系统是很多开发者贡献代码不断完善的,linux的创始人 ...