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 现在用网络流的思想就很容易解决了嘛 给什么连什么,注意是双向边,然后跑 ...
随机推荐
- 解决ie低版本不认识html5标签
在不支持HTML5新标签的浏览器里,会将这些新的标签解析成行内元素(inline)对待,所以我们只需要将其转换成块元素(block)即可使用,但是在IE9版本以下,并不能正常解析这些新标签,但是却可以 ...
- Linux/Unix下pid文件作用浅析
转载:http://blog.csdn.net/changli_90/article/details/8911191 在Linux系统的目录/var/run下面一般我们都会看到很多的*.pid文件.而 ...
- IWorkspaceFactory pWorkspaceFactory = new ShapefileWorkspaceFactoryClass(); 时,报COMException
解决办法: 在Program.cs的Main函数中添加如下代码: ESRI.ArcGIS.RuntimeManager.Bind(ESRI.ArcGIS.ProductCode.Desktop);
- 分布式锁实现方式介绍和Zookeeper实现原理
分布式锁实现的几种方式 基于数据库实现分布式锁(表.数据库排他锁) 基于缓存(redis,memcached,tair) 基于Zookeeper实现分布式锁 关注点: 单点问题?(集群) 失效时间?( ...
- 使用SQL语法来查询Elasticsearch:Elasticsearch-SQL插件
简介 Elasticsearch-SQL是Elasticsearch的一个插件,它可以让我们通过类似SQL的方式对Elasticsearch中的数据进行查询.项目地址是:https://github. ...
- Centos 修改当前路径显示为全路径
1.修改显示全路径: vim /etc/bashrc 找到[ "$PS1" = "\\s-\\v\\\$ " ] && PS1="[\ ...
- 下载xlsx文件打开一直提示文件已损坏
这是office受保护视图导致的原因所造成的,按照以下操作,问题就不是问题了 解决办法如下: 1.在打开excel2018数据表格时,出现提示“文件已损坏,无法打开”,点击确定按钮 2.进入空白程序界 ...
- JavaScript设计模式 样例二 —— 策略模式
策略模式(Strategy Pattern): 定义:定义了一族算法: 封装了每个算法: 这族的算法可互换代替. 目的:将算法的使用与算法的实现分离开来. 场景:可用来消除大量的条件分支语句. 例:J ...
- QQ群文件未通过安全检查,禁止下载该文件
直接用手机收藏群文件,然后用电脑登上qq去收藏里面下载就ok了
- 【NOIP2016提高A组模拟8.14】传送带
题目 在一个2维平面上有两条传送带,每一条传送带可以看成是一条线段.两条传送带分别为线段AB和线段CD.FTD在AB上的移动速度为P,在CD上的移动速度为Q,在平面上的移动速度R.现在FTD想从A点走 ...