bzoj 4242 水壶 (多源最短路+最小生成树+启发式合并)
4242: 水壶
Time Limit: 50 Sec Memory Limit: 512 MB
Submit: 1028 Solved: 261
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
.....
..##.
.#...
..#..
.....
1 1
4 2
3 3
2 5
1 2
2 4
1 3
3 4
Sample Output
4
4
2
HINT
跟bzoj4144题一样。。不过由于每条边距离相等,所以我们要把优先队列换成普通队列来减少时间。我实在是口算不出来这个时间是多少,然后一开始优先队列一直过不了。看了人家ac代码是普通队列一改就过了。。。orz。不过时间也是挺极限的46s。反正过啦!
#include<bits/stdc++.h>
#define clr(x) memset(x,0,sizeof(x))
#define clr_1(x) memset(x,-1,sizeof(x))
#define INF 0x3f3f3f3f
#define LL long long
#define pb push_back
#define mod 1000000007
#define ls(i) (i<<1)
#define rs(i) (i<<1|1)
#define mp make_pair
#define fi first
#define se second
using namespace std;
const int N=2e3+;
const int M=2e5+;
typedef pair<int,int> pii;
typedef pair<LL,pii> plii;
queue<pii> que;
int mart[N][N];
char s[N];
LL dis[N][N];
int pre[N][N];
bool vis[N][N];
struct edge
{
int u,v;
LL w;
edge(int _u,int _v,LL _w):u(_u),v(_v),w(_w) {}
};
vector<edge> e;
int dirx[]={,,,-},diry[]={,-,,};
void dij(int n,int m)
{
while(!que.empty())
{
pii p=que.front();
que.pop();
int x=p.fi;
int y=p.se;
LL d=dis[x][y];
if(vis[x][y]) continue;
vis[x][y]=;
for(int i=;i<;i++)
{
int a=x+dirx[i];
int b=y+diry[i];
if(mart[a][b]) continue;
if(!pre[a][b] || dis[a][b]>d+)
{
dis[a][b]=d+;
pre[a][b]=pre[x][y];
que.push(mp(a,b));
}
else if(pre[a][b]!=pre[x][y])
e.pb(edge(pre[a][b],pre[x][y],dis[x][y]+dis[a][b]));
}
}
}
int fa[M],rfa[M];
int rk[M],dep[M];
LL up[M];
bool cmp(edge a,edge b) {return a.w<b.w;}
int Find(int u) { return fa[u]==u?u:fa[u]=Find(fa[u]);}
void Union(int n)
{
sort(e.begin(),e.end(),cmp);
for(int i=;i<=n;i++)
fa[i]=i,rk[i]=;
int sz=e.size();
for(int i=;i<sz;i++)
{
int x=e[i].u;
int y=e[i].v;
LL w=e[i].w;
x=Find(x),y=Find(y);
if(x==y) continue;
if(rk[x]<rk[y]) swap(x,y);
if(rk[x]==rk[y]) rk[x]++;
fa[y]=x,rfa[y]=x,up[y]=w;
}
return ;
}
void dealdep(int u)
{
if(Find(u)==u)
{
dep[u]=;
return ;
}
dealdep(rfa[u]);
dep[u]=dep[rfa[u]]+;
return ;
}
LL solve(int u,int v)
{
LL maxn=;
if(Find(u)!=Find(v)) return -;
if(dep[u]<dep[v]) swap(u,v);
while(dep[u]>dep[v])
{
maxn=max(maxn,up[u]);
u=rfa[u];
}
while(u!=v)
{
maxn=max(maxn,up[u]);
maxn=max(maxn,up[v]);
u=rfa[u],v=rfa[v];
}
return maxn;
}
int main()
{
int n,m,k,q;
scanf("%d%d%d%d",&n,&m,&k,&q);
for(int i=;i<=n;i++)
{
scanf("%s",s+);
for(int j=;j<=m;j++)
if(s[j]=='.') mart[i][j]=;
else mart[i][j]=;
}
for(int i=;i<=m+;i++)
mart[][i]=mart[n+][i]=;
for(int i=;i<=n+;i++)
mart[i][]=mart[i][m+]=;
clr_1(dis);
for(int i=;i<=k;i++)
{
int u,v;
scanf("%d%d",&u,&v);
dis[u][v]=;
pre[u][v]=i;
que.push(mp(u,v));
}
dij(n,m);
Union(k);
for(int i=;i<=k;i++)
if(!dep[i]) dealdep(i);
// for(int i=1;i<=k;i++)
// cout<<"pos:"<<pt[i]/m+1<<" "<<pt[i]%m+1<<" blk:"<<Find(pt[i])/m+1<<" "<<Find(pt[i])%m+1<<" fa:"<<rfa[pt[i]]/m+1<<" "<<rfa[pt[i]]%m+1<<" dep:"<<dep[pt[i]]<<" val:"<<up[pt[i]]<<endl;
while(q--)
{
int u,v;
scanf("%d%d",&u,&v);
printf("%lld\n",solve(u,v));
}
return ;
}
bzoj 4242 水壶 (多源最短路+最小生成树+启发式合并)的更多相关文章
- 4144: [AMPPZ2014]Petrol (多源最短路+最小生成树+启发式合并)
4144: [AMPPZ2014]Petrol Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 752 Solved: 298[Submit][Sta ...
- BZOJ 4242: 水壶 Kruskal+BFS
4242: 水壶 Time Limit: 40 Sec Memory Limit: 256 MBSubmit: 427 Solved: 112[Submit][Status][Discuss] D ...
- BZOJ 4242 水壶(BFS建图+最小生成树+树上倍增)
题意 JOI君所居住的IOI市以一年四季都十分炎热著称. IOI市是一个被分成纵H*横W块区域的长方形,每个区域都是建筑物.原野.墙壁之一.建筑物的区域有P个,编号为1...P. JOI君只能进入建筑 ...
- BZOJ 4242: 水壶(Kruskal重构树 + Bfs)
题意 一块 \(h ∗ w\) 的区域,存在障碍.空地.\(n\) 个建筑,从一个建筑到另一个建筑的花费为:路径上最长的连续空地的长度. \(q\) 次询问:从建筑 \(s_i\) 到 \(t_i\) ...
- BZOJ 1483:[HNOI2009]梦幻布丁(链表+启发式合并)
[HNOI2009]梦幻布丁 Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2,2,1的四个布丁一 ...
- BZOJ 1483:[HNOI2009]梦幻布丁(链表启发式合并)
http://www.lydsy.com/JudgeOnline/problem.php?id=1483 题意:中文. 思路:对于每一种颜色,用一个链表串起来,一开始保存一个答案,后面颜色替换的时候再 ...
- Bzoj 2733: [HNOI2012]永无乡 数组Splay+启发式合并
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3955 Solved: 2112[Submit][Statu ...
- Bzoj 2733: [HNOI2012]永无乡(线段树+启发式合并)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MB Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己 ...
- 【BZOJ4144】[AMPPZ2014]Petrol(最短路+最小生成树+并查集)
Description 给定一个n个点.m条边的带权无向图,其中有s个点是加油站. 每辆车都有一个油量上限b,即每次行走距离不能超过b,但在加油站可以补满. q次询问,每次给出x,y,b,表示出发点是 ...
随机推荐
- C++ 和 MFC的学习
1. 在Windows应用程序设计中,既可以使用个C的基本数据类型,也可以使用Windows自定义的数据类型.但要注意,凡是Windows自定义的关键字都要大写. 2. 什么是句柄? 在Windows ...
- Linux下C程序的反汇编【转】
转自:http://blog.csdn.net/u011192270/article/details/50224267 前言:本文主要介绍几种反汇编的方法. gcc gcc的完整编译过程大致为:预处理 ...
- 遍历 USB devcie,读取设备描述符 device descriptor【转】
转自:http://blog.csdn.net/flyyyri/article/details/5480347 理论: 对于USB接口的设备,现在越来越多了.本篇我们就通过获取一个USB扫描仪设 ...
- aarch64_g5
gtkmm24-devel-2.24.5-2.fc26.aarch64.rpm 2017-02-11 18:17 620K fedora Mirroring Project gtkmm24-docs- ...
- 项目中遇到的问题:Gradle传递性依赖冲突
问题描述: 在调用别人接口时,由于他们接口做了拦截处理在使用RestTemplate调用时必须要使用@Qualifier("他们封装好的类"),需要导入jar包 gradle方式导 ...
- 【前端vue开发】vue子调父 $emit (把子组件的数据传给父组件)
ps:App.vue 父组件 Hello.vue 子组件 <!--App.vue :--> <template> <div id="app"> ...
- Nginx - upstream 模块及参数测试
目录 - 1. 前言- 2. 配置示例及指令说明 - 2.1 配置示例 - 2.2 指令 - 2.3 upstream相关变量- 3. 参数配置及测试 - 3.1 max_fa ...
- java基础78 Servlet的生命周期
1.Servlet的生命周期 简单的解析就是: 创建servlet实例(调用构造器)---->调用init()方法---->调用service()方法----->调用destroy( ...
- 回归模型效果评估系列2-MAE、MSE、RMSE、MAPE(MAPD)
MAE.MSE.RMSE.MAPE(MAPD)这些都是常见的回归预测评估指标,重温下它们的定义和区别以及优缺点吧 MAE(Mean Absolute Error) 平均绝对误差 ...
- dedecms自定义模型之独立模型在首页、列表页、内容调用内容
dedecms关于自定义模型(独立模型)的首页.列表页.内容怎么调用?在后台自定义模型(独立模型)的建立及自定义字段的添加比较简单,需要注意两点: (1)如果某个字段需要在前台列表页显示,则在前台参数 ...