duliu题之狼抓兔子题解
拖了将近5天的正解和AC.........emmmmm...........
事实告诉我们这种毒瘤题一定要建双向边(用了不知道多少个小时质疑建边的人欲哭无泪)
心态爆炸的传送
题了个面


这是个求最小割问题
说人话:
把图中的一些边砍断,使这个图分为不连通的两部分。砍断一条边的代价就是这条边的边权,求最小代价。
似乎是个定理的东西:
一个图的最小割是对偶图的最短路
question1:神马是对偶图?能吃吗?
当然不能
在这里的对偶图,就是把原来的块当做点,把原来的边建成与之垂直的边,边权不变,再在左下角和右上角新建两个点st,eend(这两个点放在哪个角上无所谓辣),作为源点和汇点,跑最短路。
为毛是eend而不是end呢?
因为在c++11下end会CE
说的太抽象了,举个例子
原图:

对偶图:

思路很简单,but代码还是很难(它是个要面子的题)
我们想想怎么给这些点编号
(其实随便编号)
窝的编号方法:
接下来我们分边讨论怎么表示点(注意一定要建双向边)
横边:

左边的红字是边的行号 i ,上边的是边的列号 j
我们要计算每条边(i,j)上面的点和下面的点,如果边的行号是1,则直接向终点eend建边,如果边的行号是 n ,就向起点st建边,否则,上下建边(注意建双向边*2)
点的表示方法:
上面的点 ss=2*(i-1)*(m-1)+j+1;
下面的点 ee=ss-m+1;
竖边:

右边的点:ss=j+(m-1)*(2*(i-1)+1)+1;
左边的点:ee=ss-m;
当j=1时:st与右边的点连边
当j=m时:左边的点与eend连边
正常情况:左边的点与右边的点连边
注意建双向边!!!(*3)
斜边:

斜上方的点:ss=2*(i-1)*(m-1)+j+1;
斜下方的点:ee=ss+m-1;
这里就不需要考虑st和eend了
双向建边*5
建完边之后跑一遍dijkstra就好辣
Code:
#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
#include<algorithm>
#define ll long long
#define pa pair<int,int>
using namespace std;
inline int read()
{
char ch=getchar(),lst;
int x=;
while(ch<''||ch>'')
{
lst=ch;
ch=getchar();
}
while(ch>=''&&ch<='')
{
x=(x<<)+(x<<)+(ch^);
ch=getchar();
}return ((lst=='-')?-x:x);
}
int n,m,st=,eend,head[],cnt,dis[];
const int inf=;
bool vis[];
struct Ed{
int to,nxt,dis;
}edge[];
void add(int fr,int to,int dis)
{
cnt++;
edge[cnt].to=to;
edge[cnt].dis=dis;
edge[cnt].nxt=head[fr];
head[fr]=cnt;
}
void dij()//堆优化的dij
{
for(int i=;i<=eend;i++)
dis[i]=inf;
dis[]=;
priority_queue<pa,vector<pa>,greater<pa> > q;
q.push(make_pair(,));
while(!q.empty())
{
int now=q.top().second;
q.pop();
if(vis[now])continue;
vis[now]=;
for(int e=head[now];e;e=edge[e].nxt)
{
int v=edge[e].to,di=edge[e].dis;
if(dis[now]+di<dis[v])
{
dis[v]=dis[now]+di;
q.push(make_pair(dis[v],v));
}
}
}
}
int main()
{
n=read();m=read();
eend=(n-)**(m-)+;
for(int i=;i<=n;i++)//横边
{
for(int j=;j<=m-;j++)
{
int dis=read();
int ss=*(i-)*(m-)+j+;
int ee=ss-m+;
if(i==)
{add(ss,eend,dis);add(eend,ss,dis);
continue;}
if(i==n)
{add(st,ee,dis);add(ee,st,dis);
continue;}
add(ss,ee,dis);add(ee,ss,dis);
}
}//竖边
for(int i=;i<n;i++)
{
for(int j=;j<=m;j++)
{
int dis=read();
int ss=j+(m-)*(*(i-)+)+;
int ee=ss-m;
if(j==)
{add(st,ss,dis);add(ss,st,dis);
continue;}
if(j==m)
{add(ee,eend,dis);add(eend,ee,dis);
continue;}
add(ee,ss,dis);add(ss,ee,dis);
}
}//斜边
for(int i=;i<n;i++)
{
for(int j=;j<m;j++)
{
int dis=read();
int ss=*(i-)*(m-)+j+;
int ee=ss+m-;
add(ee,ss,dis);
add(ss,ee,dis);
}
}
dij();
printf("%d",dis[eend]);
}
duliu题之狼抓兔子题解的更多相关文章
- 【bzoj1001】【最短路】【对偶图】【最大流转最小割】狼抓兔子题解
[BZOJ1001]狼抓兔子 1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec Memory Limit: 162 MBSubmit: 18872 Solved ...
- BZOJ1001 洛谷4001 [BJOI2006]狼抓兔子 题解
题目 这个题目有多种解法,这个题也是一个比较经典的题了,正是因为他的多样的做法,这个题主要难在建图和优化,因为这是一个网格图,所以spfa肯定过不去,所以用最短路解法的话,只能用dij,而网络流也是要 ...
- BZOJ1001 狼抓兔子 题解
裸的最小割,转化成最大流即可. #include <bits/stdc++.h> int n,m; int S,T; int mincost; int head[6001000],tot= ...
- 【BZOJ1001】狼抓兔子
1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec Memory Limit: 162 MBSubmit: 7530 Solved: 1724[Submit][S ...
- 【BZOJ1001】狼抓兔子(网络流)
[BZOJ1001]狼抓兔子(网络流) 题面 Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的, 而且现在的兔子还比较笨, ...
- BZOJ1001: [BeiJing2006]狼抓兔子【最短路+对偶图】
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1001 1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Se ...
- 【BZOJ1001】狼抓兔子(平面图转对偶图,最短路)
[BZOJ1001]狼抓兔子(平面图转对偶图,最短路) 题面 BZOJ 洛谷 题解 这题用最小割可以直接做 今天再学习了一下平面图转对偶图的做法 大致的思路如下: 1.将源点到汇点中再补一条不与任何线 ...
- BZOJ1001:狼抓兔子(最小割最大流+vector模板)
1001: [BeiJing2006]狼抓兔子 Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,而且现在的兔子还比较笨, ...
- bzoj1001: [BeiJing2006]狼抓兔子(初识是你最小割)
1001: [BeiJing2006]狼抓兔子 题目:传送门 题解: 听说这题当初是大难题...可惜当年没有网络流hahahha 现在用网络流的思想就很容易解决了嘛 给什么连什么,注意是双向边,然后跑 ...
随机推荐
- Widget代码讲解
参考:https://zhuanlan.zhihu.com/p/28225011 QT版本为5.12.4 1.main.cpp #include "widget.h" #inclu ...
- so easy(并查集+unordered_map)
There are nn points in an array with index from 11 to nn, and there are two operations to those poin ...
- [LeetCode] 108. 将有序数组转换为二叉搜索树
题目链接 : https://leetcode-cn.com/problems/convert-sorted-array-to-binary-search-tree/ 题目描述: 将一个按照升序排列的 ...
- 服务性能指标:PV、UV、TPS、QPS
名词解释 PV Page View,网页浏览量.网页被读者调用浏览的次数.网页每次打开或刷新一次页面,记录一次.用户对同一页面的多次访问,访问量累计. UV Unique Visitor,独立访问者. ...
- nodejs版实现properties后缀文件解析
1.propertiesParser.js let readline = require('readline'); let fs = require('fs'); // properties文件路径 ...
- java体系中OOP,OOD,OOA分别代表什么含义,以及OA,CRM,ERP
OOP:Object Oriented Programming 面向对象程序设计. OOD:Object Oriented Design 面向对象设计. OOA:Object Oriented Ana ...
- HTML回顾之表格
HTML表格 由什么组成? 表格由<table>标签来定义.每个表格有若干行(<tr>标签来定义),每行被分割成若干单元格(<td>标签来定义). td值表格数据, ...
- C功能模块集锦
1. offsetof #include <stddef.h> size_t offsetof(type, member); The macro offsetof() returns th ...
- idea的使用技巧
* 简介:程序员每日都会花费数小时使用ide编写和调试代码,其中很多操作都是机械重复且频率非常高,本着"工欲善其事必先利其器"的精神,闷头写代码之外花点时间研究一下自己用的ide, ...
- CRMEasy知识库访问权限
知识库设置BCCBINFO文件夹为共享文件夹,给予普通域用户和管理员域用户完全控制权限,但是当以管理员域用户身份对文件进行操作后,以普通域用户登录不会有效果,甚至出现“访问权限”的错误,这是由于文件夹 ...