【bzoj1001】[BeiJing2006]狼抓兔子 最小割+对偶图+最短路
题目描述
现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形:
左上角点为(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只狼,才能完全封锁这条道路,你需要帮助狼王安排一个伏击方案,使得在将兔子一网打尽的前提下,参与的狼的数量要最小。因为狼还要去找喜羊羊麻烦.
输入
第一行为N,M.表示网格的大小,N,M均小于等于1000.
接下来分三部分
第一部分共N行,每行M-1个数,表示横向道路的权值.
第二部分共N-1行,每行M个数,表示纵向道路的权值.
第三部分共N-1行,每行M-1个数,表示斜向道路的权值.
输入文件保证不超过10M
输出
输出一个整数,表示参与伏击的狼的最小数量.
样例输入
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
样例输出
14
题解
最小割,转化成对偶图最短路来求。
由于点数边数都很大,直接跑最大流肯定会TLE。
想到题目中图有特殊规律,方便转化为对偶图。
于是可以先转化为对偶图,再求最短路。
步骤:
1.连一条s->t的边
2.为图中每个面积块标号,方法自己选择,s->t边内侧为(s'),外侧为(t')(反过来也一样,因为无向图)
3.连接题目中每条边挨着的两个面积块,权值为原边权,注意要连无向边。
效果:
其中黑色为原图边,红色为新点,蓝色为新边,蓝色数字为新边权。
看似很麻烦,点边很多,实际上堆优化Dijkstra很快,而Dinic慢到死。
然后跑堆优化Dijkstra即可。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 | #include <cstdio> #include <cstring> #include <utility> #include <queue> using namespace std; priority_queue<pair< int , int > > q; int head[2000010] , to[6000010] , len[6000010] , next[6000010] , cnt , dis[2000010] , vis[2000010]; void add( int x , int y , int z) { to[++cnt] = y; len[cnt] = z; next[cnt] = head[x]; head[x] = cnt; } int main() { int n , m , i , j , x , y , z , s , t; scanf ( "%d%d" , &n , &m); s = 0 , t = (n - 1) * (m - 1) * 2 + 1; for (i = 1 ; i <= n ; i ++ ) { for (j = 1 ; j < m ; j ++ ) { scanf ( "%d" , &z); if (i == 1) x = s; else x = (i - 2) * (m - 1) * 2 + (j - 1) * 2 + 1; if (i == n) y = t; else y = (i - 1) * (m - 1) * 2 + (j - 1) * 2 + 2; add(x , y , z) , add(y , x , z); } } for (i = 1 ; i < n ; i ++ ) { for (j = 1 ; j <= m ; j ++ ) { scanf ( "%d" , &z); if (j == 1) x = t; else x = (i - 1) * (m - 1) * 2 + (j - 2) * 2 + 2; if (j == m) y = s; else y = (i - 1) * (m - 1) * 2 + (j - 1) * 2 + 1; add(x , y , z) , add(y , x , z); } } for (i = 1 ; i < n ; i ++ ) { for (j = 1 ; j < m ; j ++ ) { scanf ( "%d" , &z); x = (i - 1) * (m - 1) * 2 + (j - 1) * 2 + 1; y = (i - 1) * (m - 1) * 2 + (j - 1) * 2 + 2; add(x , y , z) , add(y , x , z); } } memset (dis , 0x3f , sizeof (dis)); dis[s] = 0; q.push(make_pair(0 , s)); while (!q.empty()) { x = q.top().second , q.pop(); if (vis[x]) continue ; vis[x] = 1; for (i = head[x] ; i ; i = next[i]) if (dis[to[i]] > dis[x] + len[i]) dis[to[i]] = dis[x] + len[i] , q.push(make_pair(-dis[to[i]] , to[i])); } printf ( "%d\n" , dis[t]); return 0; } |
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步