BZOJ4356Ceoi2014 Wall——堆优化dijkstra+最短路树
题目描述
输入
输出
样例输入
3 3
1 0 0
1 0 0
0 0 1
1 4 9 4
1 6 6 6
1 2 2 9
1 1 1
4 4 4
2 4 2
6 6 6
input 2
3 3
1 0 1
0 0 0
0 1 0
2 1 1 3
5 6 1 1
2 1 1 3
2 1 1
3 4 1
4 1 1
5 1 2
样例输出
38
output 2
22
提示
首先我们求出首都(即左上角)到每个城市左上角那个点的最短路形成的最短路树,那么有一个结论:所选的环一定将最短路树上的边圈在里面。
因为如果有一些最短路的边在环外,那么将环被这段最短路包含的部分换成这段最短路答案一定不会变劣。
所以我们现在就是要求在环不穿过城市和最短路树上的边的情况下最小。
我们将原图的每个点拆成四个点,如图所示。
即将每个点拆成左上、左下、右上、右下四个点。
对于每个原图点拆成的四个点中每对相邻点,只要他们两个之间的边不是城市边或最短路树上的边,那么就将他们两个连一条边权为$0$的边。
特别注意的是原图左上角那个点拆成的$1$号点不与$2,3$号点连边(原因下面会提到)。
对于原图每条边两侧的两对点(例如$A2,B1$和$A4,B3$)分别将他们连边,边权为原图这条边的边权。
这样我们跑出从左上角$3$号点到$2$号点的最短路即为答案。
因为连边时不穿过城市边和最短路树上的边,且起点保证在这些边的外围,所以能将所有城市都圈在环内。
而从$3$号点跑最短路到$2$号点保证它是一个环且最短。不允许穿过最短路树边相当于将每个城市与起点连通。
#include<set>
#include<map>
#include<cmath>
#include<queue>
#include<stack>
#include<vector>
#include<cstdio>
#include<bitset>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
#define pr pair<ll,int>
#define PR pair<int,int>
using namespace std;
int vis[410][410];
int num;
int sum;
int n,m;
int head[650000];
ll d[650000];
int to[5200000];
int next[5200000];
int val[5200000];
int v[650000];
int w[410][410][5];
int tot;
int x,y;
int mp[400000];
inline void add(int x,int y,int z)
{
next[++tot]=head[x];
head[x]=tot;
to[tot]=y;
val[tot]=z;
}
inline int get(int x,int y)
{
x++,y++;
return (x-1)*(m+1)+y;
}
inline void back(int num,int &x,int &y)
{
x=(num-1)/(m+1)+1;
y=num-(m+1)*(x-1);
x--,y--;
}
namespace build
{
int head[170000];
ll d[170000];
int from[170000];
int vis[170000];
int to[680000];
int next[680000];
int val[680000];
int tot;
inline void add(int x,int y,int z)
{
next[++tot]=head[x];
head[x]=tot;
to[tot]=y;
val[tot]=z;
}
inline void dijkstra()
{
priority_queue< pr,vector<pr>,greater<pr> >q;
for(int i=1;i<=num;i++)
{
d[i]=1ll<<60;
}
d[1]=0;
q.push(make_pair(d[1],1));
while(!q.empty())
{
int x=q.top().second;
q.pop();
if(vis[x])
{
continue;
}
vis[x]=1;
for(int i=head[x];i;i=next[i])
{
if(d[to[i]]>d[x]+1ll*val[i])
{
d[to[i]]=d[x]+1ll*val[i];
from[to[i]]=x;
q.push(make_pair(d[to[i]],to[i]));
}
}
}
}
}
inline void dijkstra()
{
priority_queue< pr,vector<pr>,greater<pr> >q;
for(int i=1;i<=num*4;i++)
{
d[i]=1ll<<60;
}
d[num+1]=0;
q.push(make_pair(d[num+1],num+1));
while(!q.empty())
{
int now=q.top().second;
q.pop();
if(v[now])
{
continue;
}
v[now]=1;
for(int i=head[now];i;i=next[i])
{
if(d[to[i]]>d[now]+1ll*val[i])
{
d[to[i]]=d[now]+1ll*val[i];
q.push(make_pair(d[to[i]],to[i]));
}
}
}
printf("%lld",d[num*2+1]);
}
inline int find(int x,int y)
{
if(x>y)
{
swap(x,y);
}
int a,b,c,d;
back(x,a,b);
back(y,c,d);
if(b==d)
{
return (c-1)*(m+1)+d+1;
}
else
{
return sum+c*m+d;
}
}
int main()
{
scanf("%d%d",&n,&m);
num=(n+1)*(m+1);
sum=n*(m+1);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%d",&vis[i][j]);
}
}
for(int i=1;i<=n;i++)
{
for(int j=0;j<=m;j++)
{
scanf("%d",&x);
build::add(get(i-1,j),get(i,j),x);
build::add(get(i,j),get(i-1,j),x);
w[i-1][j][3]=w[i][j][1]=x;
}
}
for(int i=0;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%d",&x);
build::add(get(i,j-1),get(i,j),x);
build::add(get(i,j),get(i,j-1),x);
w[i][j-1][2]=w[i][j][4]=x;
}
}
build::dijkstra();
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
if(!vis[i-1][j-1]&&!vis[i-1][j]&&!vis[i][j-1]&&vis[i][j])
{
for(x=get(i-1,j-1);x!=1;x=build::from[x])
{
mp[find(build::from[x],x)]=1;
}
}
}
}
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
if(vis[i][j])
{
mp[find(get(i,j),get(i-1,j))]=1;
mp[find(get(i,j),get(i,j-1))]=1;
mp[find(get(i-1,j-1),get(i,j-1))]=1;
mp[find(get(i-1,j-1),get(i-1,j))]=1;
}
}
}
for(int i=1;i<=num;i++)
{
back(i,x,y);
if((!y||!mp[find(get(x,y),get(x,y-1))])&&i!=1)
{
add(i,2*num+i,0);
add(2*num+i,i,0);
}
if((!x||!mp[find(get(x,y),get(x-1,y))])&&i!=1)
{
add(i,num+i,0);
add(num+i,i,0);
}
if(x==n||!mp[find(get(x,y),get(x+1,y))])
{
add(num*2+i,num*3+i,0);
add(num*3+i,num*2+i,0);
}
if(y==m||!mp[find(get(x,y),get(x,y+1))])
{
add(num+i,num*3+i,0);
add(num*3+i,num+i,0);
}
if(y)
{
add(i,num+get(x,y-1),w[x][y][4]);
add(num*2+i,num*3+get(x,y-1),w[x][y][4]);
}
if(x)
{
add(i,num*2+get(x-1,y),w[x][y][1]);
add(num+i,num*3+get(x-1,y),w[x][y][1]);
}
if(x!=n)
{
add(num*2+i,get(x+1,y),w[x][y][3]);
add(num*3+i,num+get(x+1,y),w[x][y][3]);
}
if(y!=m)
{
add(num+i,get(x,y+1),w[x][y][2]);
add(num*3+i,num*2+get(x,y+1),w[x][y][2]);
}
}
dijkstra();
}
BZOJ4356Ceoi2014 Wall——堆优化dijkstra+最短路树的更多相关文章
- [CF1051F]The Shortest Statement_堆优化dij_最短路树_倍增lca
The Shortest Statement 题目链接:https://codeforces.com/contest/1051/problem/F 数据范围:略. 题解: 关于这个题,有一个重要的性质 ...
- 【bzoj1922】[Sdoi2010]大陆争霸 堆优化Dijkstra
题目描述 一张n个点m条边的图,通过每条边需要一定的时间.有一些限制条件,每个限制条件形如“x保护y”,表示到达y的最短时间不能小于到达x的最短时间(即如果在其之前到达,则需要等待至xd到达).问1到 ...
- 【bzoj1097】[POI2007]旅游景点atr 状压dp+堆优化Dijkstra
题目描述 FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个 ...
- 【bzoj4016】[FJOI2014]最短路径树问题 堆优化Dijkstra+DFS树+树的点分治
题目描述 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多条长度最短的路径,则选择经过的顶点序列字典序最小的那条路径( ...
- 洛谷 P3371 【模板】单源最短路径(堆优化dijkstra)
题目描述 如题,给出一个有向图,请输出从某一点出发到所有点的最短路径长度. 输入输出格式 输入格式: 第一行包含三个整数N.M.S,分别表示点的个数.有向边的个数.出发点的编号. 接下来M行每行包含三 ...
- PAT-1030 Travel Plan (30 分) 最短路最小边权 堆优化dijkstra+DFS
PAT 1030 最短路最小边权 堆优化dijkstra+DFS 1030 Travel Plan (30 分) A traveler's map gives the distances betwee ...
- 堆优化 dijkstra 简介
dijkstra 前言 原本我真的不会什么 dijkstra 只用那已死的 spfa ,还有各种玄学优化,可是,我不能相信一个已死的算法,就像我不能相信自己. ps : 虽然他已经活了 序 我站在镜子 ...
- BZOJ 3040 最短路 (堆优化dijkstra)
这题不是裸的最短路么?但是一看数据范围就傻了.点数10^6,边数10^7.这个spfa就别想了(本来spfa就是相当不靠谱的玩意),看来是要用堆优化dijkstra了.但是,平时写dijkstra时为 ...
- UVA - 11374 - Airport Express(堆优化Dijkstra)
Problem UVA - 11374 - Airport Express Time Limit: 1000 mSec Problem Description In a small city c ...
随机推荐
- wpf、winform仿QQ靠边隐藏
先说下下面的代码和demo是wpf的,如果winform要用,改动不大的. 实现思路: 通过定时刷新鼠标位置 和 窗体坐标 进行计算 来控制窗体的隐藏 显示 代码都有详细的注释 //窗体状态 true ...
- 【原创】研发应该懂的binlog知识(上)
引言 为什么写这篇文章? 大家当年在学MySQL的时候,为了能够迅速就业,一般是学习一下MySQL的基本语法,差不多就出山找工作了.水平稍微好一点的童鞋呢还会懂一点存储过程的编写,又或者是懂一点索引的 ...
- Asp.Net Core实战(干货)
序言 使用.NET Core,团队可以更容易专注的在.net core上工作.比如核心类库(如System.Collections)的更改仍然需要与.NET Framework相同的活力,但是ASP. ...
- WCF系列教程之消息交换模式之请求与答复模式(Request/Reply)
1.使用WCF请求与答复模式须知 (1).客户端调用WCF服务端需要等待服务端的返回,即使返回类型是void (2).相比Duplex来讲,这种模式强调的是客户端的被动接受,也就是说客户端接受到响应后 ...
- 第四次oo博客
论述测试与正确性论证的效果差异 单元测试利用测试者构造的测试用例来检查类或方法的正确性,一般来说所需要测试的用例是无穷多的,通过人为构造代表性的测试用例来尽量测试所有代码.测试的优点在于不易出错,只要 ...
- 关于jsp中jstl-core标签循环遍历的使用
JSP标准标签库(JSTL)是一个JSP标签集合,它封装了JSP应用的通用核心功能. JSTL支持通用的.结构化的任务,比如迭代,条件判断,XML文档操作,国际化标签,SQL标签. 除了这些,它还提供 ...
- UnderWater+SDN论文之五
Underwater Sensor Networks with Mobile Agents: Experience from the Field Source: LNICST 2013 论文是来自两个 ...
- Hadoop生态的配置
网盘下载地址 链接: https://pan.baidu.com/s/19qWnP6LQ-cHVrvT0o1jTMg 密码: 44hs Hadoop伪分布式配置 Hadoop 可以在单节点上以伪分布 ...
- PEP 8 python编程规范
一 代码编排 缩进.4个空格的缩进(编辑器都可以完成此功能),不使用Tap,更不能混合使用Tap和空格. 每行最大长度79,换行可以使用反斜杠,最好使用圆括号.换行点要在操作符的后边敲回车. 类和to ...
- Telnet服务器和客户端请求处理
Telnet服务器和客户端请求处理 本文的控制台项目是根据SuperSocket官方Telnet示例代码进行调试的,官方示例代码:Telnet示例. 开始我的第一个Telnet控制台项目之旅: 创建控 ...