【NOI2010】海拔【平面图最小割】
【问题描写叙述】
YT市是一个规划良好的城市,城市被东西向和南北向的主干道划分为n×n个区域。简单起见。能够将YT市看作 一个正方形,每个区域也可看作一个正方形。从而。YT城市中包含(n+1)×(n+1)个交叉路口和2n×(n+1)条双向道路(简称道路),每条双向 道路连接主干道上两个相邻的交叉路口。下图为一张YT市的地图(n = 2),城市被划分为2×2个区域,包含3×3个交叉路口和12条双向道路。
小Z作为该市的市长。他依据统计信息得到了每天上班高峰期间YT市每条道路两个方向的人流量。即在高峰期间沿 着该方向通过这条道路的人数。每个交叉路口都有不同的海拔高度值,YT市市民觉得爬坡是一件很累的事情,每向上爬h的高度,就须要消耗h的体力。
假设 是下坡的话,则不须要耗费体力。
因此假设一段道路的终点海拔减去起点海拔的值为h(注意h可能是负数)。那么一个人经过这段路所消耗的体力是max{0, h}(这里max{a, b}表示取a, b两个值中的较大值)。
小Z还測量得到这个城市西北角的交叉路口海拔为0。东南角的交叉路口海拔为1(如上图所看到的)。但其他交叉路口的海拔高度都无法得知。小Z想知道在最理想的情况下(即你能够随意如果其他路口的海拔高度)。每天上班高峰期间全部人爬坡消耗的整体力和的最小值。
【输入格式】
第一行包括一个整数n。含义如上文所看到的。
接下来4n(n + 1)行,每行包括一个非负整数分别表示每一条道路每个方向的人流量信息。输入顺序:n(n + 1)个数表示全部从西到东方向的人流量,然后n(n + 1)个数表示全部从北到南方向的人流量,n(n + 1)个数表示全部从东到西方向的人流量。最后是n(n + 1)个数表示全部从南到北方向的人流量。对于每个方向,输入顺序依照起点由北向南,若南北方向同样时由西到东的顺序给出(參见例子输入)。
【输出格式】
仅包括一个数,表示在最理想情况下每天上班高峰期间全部人爬坡所消耗的整体力和(即整体力和的最小值),结果四舍五入到整数。
【例子输入】
1
1
2
3
4
5
6
7
8
【例子输出】
3
【例子说明】
例子数据见下图。
最理想情况下全部点的海拔如上图所看到的。
【数据规模】
对于20%的数据:n ≤ 3。
对于50%的数据:n ≤ 15;
对于80%的数据:n ≤ 40。
对于100%的数据:1 ≤ n ≤ 500,0 ≤ 流量 ≤ 1,000,000且全部流量均为整数。
【提示】
海拔高度不一定是整数。
【执行时限】
2秒。
【执行空限】
512M。
题解:首先依据调整法能够证明高度中仅仅存在0和1。而且0一定连在一起,1一定连在一起,然后就能够想到求最小割,由于数据范围比較大。我们须要转化成对偶图求最短路。另一点问题是这个图是有向图。解决办法是把每条边都逆时针转90度。
(普通spfa会超时两个点,须要使用slf优化或堆优化dijkstra)
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int s,t,c,n,point[5000001],next[5000001],cnt,dis[5000001],l[50000001];
bool f[5000001];
struct use{
int st,en,val;
}b[5000001];
void add(int x,int y,int z)
{
next[++cnt]=point[x];point[x]=cnt;
b[cnt].st=x;b[cnt].en=y;b[cnt].val=z;
}
int spfa(int x,int y)
{
int h,t,u;
memset(dis,127/3,sizeof(dis));
dis[x]=0;h=0;t=1;l[t]=x;f[x]=true;
while (h<t)
{
u=l[++h];
f[u]=false;
for (int i=point[u];i;i=next[i])
if (dis[b[i].en]>dis[u]+b[i].val&&b[i].en!=u)
{
dis[b[i].en]=dis[u]+b[i].val;
if (!f[b[i].en])
{
f[b[i].en]=true;
if (dis[b[i].en]<dis[l[h+1]]) l[h--]=b[i].en;
else
l[++t]=b[i].en;
}
}
}
if (dis[y]>510000000) dis[y]=0;
return dis[y];
}
int main()
{
freopen("altitude.in","r",stdin);
freopen("altitude.out","w",stdout);
scanf("%d",&n);
s=1;t=n*n+2;
for (int i=1;i<=n+1;i++)
for (int j=1;j<=n;j++)
{
scanf("%d",&c);
if (i==1) add(j+1,t,c);
if (i==n+1) add(s,n*(n-1)+j+1,c);
if (i>1&&i<n+1) add(n*(i-1)+j+1,n*(i-2)+j+1,c);
}
for (int i=1;i<=n;i++)
for (int j=1;j<=n+1;j++)
{
scanf("%d",&c);
if (j==1) add(s,n*(i-1)+2,c);
if (j==n+1) add(n*(i-1)+n+1,t,c);
if (j>1&&j<n+1) add(n*(i-1)+j,n*(i-1)+j+1,c);
}
for (int i=1;i<=n+1;i++)
for (int j=1;j<=n;j++)
{
scanf("%d",&c);
if (i==1) add(t,j+1,c);
if (i==n+1) add(n*(n-1)+j+1,s,c);
if (i>1&&i<n+1) add(n*(i-2)+j+1,n*(i-1)+j+1,c);
}
for (int i=1;i<=n;i++)
for (int j=1;j<=n+1;j++)
{
scanf("%d",&c);
if (j==1) add(n*(i-1)+2,s,c);
if (j==n+1) add(t,n*(i-1)+n+1,c);
if (j>1&&j<n+1) add(n*(i-1)+j+1,n*(i-1)+j,c);
}
cout<<spfa(s,t);
}
【NOI2010】海拔【平面图最小割】的更多相关文章
- Vijos1734 NOI2010 海拔 平面图最小割
建立平面图的对偶图,把最小割转化成最短路问题 Dijkstra算法堆优化 (被输入顺序搞WA了好几次T_T) #include <cstdio> #include <cstring& ...
- bzoj2007/luoguP2046 海拔(平面图最小割转对偶图最短路)
bzoj2007/luoguP2046 海拔(平面图最小割转对偶图最短路) 题目描述: bzoj luogu 题解时间: 首先考虑海拔待定点的$h$都应该是多少 很明显它们都是$0$或$1$,并且所 ...
- BZOJ2007/LG2046 「NOI2010」海拔 平面图最小割转对偶图最短路
问题描述 BZOJ2007 LG2046 题解 发现左上角海拔为 \(0\) ,右上角海拔为 \(1\) . 上坡要付出代价,下坡没有收益,所以有坡度的路越少越好. 所以海拔为 \(1\) 的点,和海 ...
- BZOJ 2007 海拔(平面图最小割转对偶图最短路)
首先注意到,把一个点的海拔定为>1的数是毫无意义的.实际上,可以转化为把这些点的海拔要么定为0,要么定为1. 其次,如果一个点周围的点的海拔没有和它相同的,那么这个点的海拔也是可以优化的,即把这 ...
- 洛谷P2046 [NOI2010]海拔(最小割,平面图转对偶图)
传送门 不明白为什么大佬们一眼就看出这是最小割…… 所以总而言之这就是一个最小割我也不知道为什么 然后边数太多直接跑会炸,所以要把平面图转对偶图,然后跑一个最短路即可 至于建图……请看代码我实在无能为 ...
- bzoj 2007: [Noi2010]海拔【最小割+dijskstra】
上来就跑3e5的最大流--脑子抽了 很容易看出,每个地方的海拔都是0或1因为再高了没有意义,又,上去下来再上去没有意义,所以最后一定是从s连着一片0,剩下连着t一片1,然后有贡献的就是01交接的那些边 ...
- B20J_2007_[Noi2010]海拔_平面图最小割转对偶图+堆优化Dij
B20J_2007_[Noi2010]海拔_平面图最小割转对偶图+堆优化Dij 题意:城市被东西向和南北向的主干道划分为n×n个区域.城市中包括(n+1)×(n+1)个交叉路口和2n×(n+1)条双向 ...
- 【BZOJ2007】【NOI2010】海拔(最小割,平面图转对偶图,最短路)
[BZOJ2007][NOI2010]海拔(最小割,平面图转对偶图,最短路) 题面 BZOJ 洛谷 Description YT市是一个规划良好的城市,城市被东西向和南北向的主干道划分为n×n个区域. ...
- [BZOJ 2007] [Noi2010] 海拔 【平面图最小割(对偶图最短路)】
题目链接:BZOJ - 2007 题目分析 首先,左上角的高度是 0 ,右下角的高度是 1.那么所有点的高度一定要在 0 与 1 之间.然而选取 [0, 1] 的任何一个实数,都可以用整数 0 或 1 ...
- Luogu2046 NOI2010 海拔 平面图、最小割、最短路
传送门 首先一个不知道怎么证的结论:任意点的\(H\)只会是\(0\)或\(1\) 那么可以发现原题的本质就是一个最小割,左上角为\(S\),右下角为\(T\),被割开的两个部分就是\(H=0\)与\ ...
随机推荐
- Flex 基础语法(一)
任何一个容器都可以指定为Flex布局. .box{ display: flex; } 行内元素也可以使用Flex布局. .box{ display: inline-flex; } Webkit内核的浏 ...
- ueditor插入百度音乐无法播放-403 问题
简单记录一下,其实403是因为百度音乐设置了禁止外部连接引用,因此 几乎所有的百度音乐播放都会提示403. 注意:预览页面(dialog/music/music.html)和实际插入页面(uedito ...
- 数据库中float类型字段,转化到前端显示,统一保留两位小数
客户的一个需求,mybatis查询到的数据库的数据进行转换,采用TypeHandler<T>的方式.float保留两位精度可以采用DecimalFormat 直接贴上最终的解决代码(事情没 ...
- div内长串数字或字母不断行处理
比如: <div>1111tryrt645645rt4554111112324353453454364</div> <div>qwewretrytuytuiyiuo ...
- 让C++控制台程序停下来,实现暂停功能
一.针对Microsoft #include <stdlib.h> (1)第一种方式system( "PAUSE "); -------------------- ...
- Java多线程学习之线程池源码详解
0.使用线程池的必要性 在生产环境中,如果为每个任务分配一个线程,会造成许多问题: 线程生命周期的开销非常高.线程的创建和销毁都要付出代价.比如,线程的创建需要时间,延迟处理请求.如果请求的到达率非常 ...
- linux操作系统基础篇(二)
Linux用户.群组.权限 1.用户也是由一个个文件组成的下列文件都是存放用户信息的文件 useradd user1 /etc/passwd: 存放用户信息 /etc/shadow/ :存放用户密码 ...
- PullToRefreshListView插件初次进入页面自动刷新
只要将PullToRefreshListView源码中的: @Override protected void onRefreshing(final boolean doScroll) { /** * ...
- Redis学习-内存优化
以下为个人学习Redis的备忘录--内存优化 1.随时查看info memory,了解内存使用状况:127.0.0.1:6379> info memory# Memoryused_memory: ...
- Centos6.9安装vsftpd并配置多用户的方法
本文介绍了Centos6.9安装vsftpd并配置多用户的方法,分享给大家,具体如下: 一.安装vsftpd ? 1 2 3 4 5 6 7 8 #安装vsftpd yum -y install vs ...