拖了将近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题之狼抓兔子题解的更多相关文章

  1. 【bzoj1001】【最短路】【对偶图】【最大流转最小割】狼抓兔子题解

    [BZOJ1001]狼抓兔子 1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 18872  Solved ...

  2. BZOJ1001 洛谷4001 [BJOI2006]狼抓兔子 题解

    题目 这个题目有多种解法,这个题也是一个比较经典的题了,正是因为他的多样的做法,这个题主要难在建图和优化,因为这是一个网格图,所以spfa肯定过不去,所以用最短路解法的话,只能用dij,而网络流也是要 ...

  3. BZOJ1001 狼抓兔子 题解

    裸的最小割,转化成最大流即可. #include <bits/stdc++.h> int n,m; int S,T; int mincost; int head[6001000],tot= ...

  4. 【BZOJ1001】狼抓兔子

    1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 7530  Solved: 1724[Submit][S ...

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

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

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

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

  7. 【BZOJ1001】狼抓兔子(平面图转对偶图,最短路)

    [BZOJ1001]狼抓兔子(平面图转对偶图,最短路) 题面 BZOJ 洛谷 题解 这题用最小割可以直接做 今天再学习了一下平面图转对偶图的做法 大致的思路如下: 1.将源点到汇点中再补一条不与任何线 ...

  8. BZOJ1001:狼抓兔子(最小割最大流+vector模板)

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

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

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

随机推荐

  1. P1067多项式输出

    这道题是2009普及组的题,仍然是一个字符串+模拟.(蒻到先不刷算法) 这道题的题干给了很多的提示,也很全面,但是当我把种种情况都考虑到了后,在写代码的过程中仍然出现了很多的错误,wa了三四次.其实导 ...

  2. 通过编写串口助手工具学习MFC过程——(二)通过“打开串口”按钮了解基本操作

    通过编写串口助手工具学习MFC过程 因为以前也做过几次MFC的编程,每次都是项目完成时,MFC基本操作清楚了,但是过好长时间不再接触MFC的项目,再次做MFC的项目时,又要从头开始熟悉.这次通过做一个 ...

  3. BZOJ 5317: [Jsoi2018]部落战争

    传送门 写出式子,若存在 $a \in A$,$b \in B$,使得 $b+v=a$,那么此方案会产生冲突 即存在 $a \in A$,$b \in B$,使得 $v=a+(-b)$,设 $C=A+ ...

  4. TMS320F28335——SCI串口

    一.IO配置 以SCIA为例:使用的是GPIO35--SCITXDA    GPIOA36--SCIRXDA 使用寄存器: GPBPUD :设置上拉  GPIO32-GPIO63   对应位0 使能上 ...

  5. Java 关于SpringbootJPA分页及排序

    创建Pageable对象 看了网上很多博客,都是在用 new PageRequest的方法创建Pageable对象.可是估计很多同学写了之后才发现原来这个方法作者已经标记为过时了: 替代的方法是不要n ...

  6. memset,内存初始化函数

    # include <string.h> void *memset(void *s, int c, unsigned long n); 函数的功能是:将指针变量 s 所指向的前 n 字节的 ...

  7. 企业微信的corpsecret在哪里?

      问题: 查看“企业微信”的官方开发文档,在“获取access_token”部分提到,使用GET请求方法,访问 https://qyapi.weixin.qq.com/cgi-bin/gettoke ...

  8. 005-sort 命令使用

    字符处理命令 sort 排序命令 排序选项: -f, --ignore-case 忽略字母大小写 -n, --numeric-sort 根据字符串数值比较 -r, --reverse 逆序输出排序结果 ...

  9. 09java进阶——IO

    1.File类 1.1目录及路径分隔符 package cn.jxufe.java.chapter09.demo01; import java.io.File; public class Test01 ...

  10. 苹果ios开发

    苹果上架:https://blog.csdn.net/pcf1995/article/details/79650345