狼抓兔子 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. constraint和index--转载

    primary key和unique约束是要依赖index的,下面通过试验来看看他们之间的依赖关系!       SQL> select * from tt;   ID NA --------- ...

  2. 关于每次取PC的值为PC+4的问题

    关于ARM的书上常说由于流水线特性,在指令执行期间读取程序计数器时,读出的值需要为当前指令+4 一开始总是不理解,今天被一位大神一语道破其中精髓.... 程序计数器(PC)总是指向“正在取指”的指令 ...

  3. 【POJ2248、LOJ#10021】 Addition Chains

    事先预警:由于我太蒻了,本做法只能在POJ.LOJ等小数据(N<=100)平台上通过,在UVa(洛谷)上大数据并不能通过 戳我获得更好的观看效果 本题不用看,爆搜就是了,但是纯爆搜显然会爆时间, ...

  4. .Net Core添加分布式Session

    一.Session HTTP是一个无状态协议,Web服务器将每一个请求都视为独立请求.并且不保存之前请求中用户的值. Session 状态是ASP.NET Core提供的一个功能,它可以在用户通应用访 ...

  5. Java中final、finally、finalize的区别与用法

    1.简单区别:final用于声明属性,方法和类,分别表示属性不可交变,方法不可覆盖,类不可继承.finally是异常处理语句结构的一部分,表示总是执行.finalize是Object类的一个方法,在垃 ...

  6. Android Retrofit 2.0文件上传

    Android Retrofit 实现(图文上传)文字(参数)和多张图片一起上传 使用Retrofit进行文件上传,肯定离不开Part & PartMap. public interface ...

  7. mysql 导入数据库时,报错1840的解决方法

    1.现象 在mysql用sql文件导入数据库时,提示ERROR 1840 (HY000) at line 24: @@GLOBAL.GTID_PURGED can only be set when @ ...

  8. SQL Server2008优化之SET STATISTICS开关

    一.准备工作 缓存对于某个查询的性能影响十分之大,所以优化之前要清空缓存. 清除Buffer Pool时面的所有缓存 DBCC DROPCLEANBUFFERS清除Buffer Pool里的所有缓存的 ...

  9. ★Java语法(一)——————————标识符

    1.定义:用来标识类名.变量名.方法名.数组名.文件名的有效字符序列: 2.命名规则:a 由字母.数字._(下划线)和$(美元符号)组成 b 数字不能作为第一个字符 c 不能是Java中的关键字和保留 ...

  10. Oracle存储过程给变量赋值的方法

    截止到目前我发现有三种方法可以在存储过程中给变量进行赋值: 1.直接法     := 如:v_flag := 0; 2.select into 如:假设变量名为v_flag,select count( ...