BZOJ 4144 Dijkstra+Kruskal+倍增LCA
思路:
先把所有的加油站 push进按weight排序的优先队列里
对于每个不是加油站的点 找到到它的点的最短路以及它来源的加油站
如果x和y有边 且x和y加油站的来源不一样 则它可以连边
跑一边Kruskal
倍增查一下 搞定了
(注意图可能不连通)
//By SiriusRen
#include <queue>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 444444
int n,s,m,Station[N],rec[N],cnt,f[N],fa[N][35],maxx[N][35],T;
int q,xx,yy,zz,w[N],v[N],first[N],next[N],tot,vis[N],dis[N],deep[N];
struct Node{
int from,now,weight;
Node(){}
Node(int x,int y,int z){
from=x,now=y,weight=z;
}
}jy,node[N];
priority_queue<Node>pq;
bool operator < (Node a,Node b){
return a.weight>b.weight;
}
void add(int x,int y,int z){
w[tot]=z,v[tot]=y,next[tot]=first[x],first[x]=tot++;
}
void Dijkstra(){
while(!pq.empty()){
Node t=pq.top();pq.pop();
if(vis[t.now])continue;
vis[t.now]=1;
for(int i=first[t.now];~i;i=next[i])
if(dis[v[i]]>dis[t.now]+w[i]){
dis[v[i]]=dis[t.now]+w[i];
rec[v[i]]=t.from;
pq.push(Node(t.from,v[i],dis[v[i]]));
}
}
}
int W[N],V[N],NEXT[N],FIRST[N],TOT;
void add2(int X,int Y,int Z){
W[TOT]=Z,V[TOT]=Y,NEXT[TOT]=FIRST[X],FIRST[X]=TOT++;
}
int find(int x){return x==f[x]?x:f[x]=find(f[x]);}
void Kruskal(){
for(int i=1;i<=n;i++)f[i]=i;
sort(node+1,node+1+cnt);
for(int i=cnt;i;i--){
int fx=find(node[i].now),fy=find(node[i].from);
if(fx!=fy){
f[fx]=fy;
add2(node[i].now,node[i].from,node[i].weight);
add2(node[i].from,node[i].now,node[i].weight);
}
}
}
void dfs(int x,int father){
vis[x]=T;
for(int i=1;i<=29;i++){
fa[x][i]=fa[fa[x][i-1]][i-1];
maxx[x][i]=max(maxx[x][i-1],maxx[fa[x][i-1]][i-1]);
}
for(int i=FIRST[x];~i;i=NEXT[i])
if(V[i]!=father){
deep[V[i]]=deep[x]+1;
fa[V[i]][0]=x,maxx[V[i]][0]=W[i];
dfs(V[i],x);
}
}
int lca(int x,int y){
int ans=0;
if(deep[x]<deep[y])swap(x,y);
for(int i=29;~i;i--)
if(deep[x]-(1<<i)>=deep[y])
ans=max(ans,maxx[x][i]);x=fa[x][i];
if(x==y)return ans;
for(int i=29;~i;i--)
if(fa[x][i]!=fa[y][i]){
ans=max(ans,max(maxx[x][i],maxx[y][i]));
x=fa[x][i],y=fa[y][i];
}
return max(ans,max(maxx[x][0],maxx[y][0]));
}
int main(){
memset(FIRST,-1,sizeof(FIRST));
memset(first,-1,sizeof(first));
memset(dis,0x3f,sizeof(dis));
scanf("%d%d%d",&n,&s,&m);
for(int i=1;i<=s;i++){
scanf("%d",&xx);
dis[xx]=0,Station[xx]=1,rec[xx]=xx;
pq.push(Node(xx,xx,0));
}
for(int i=1;i<=m;i++){
scanf("%d%d%d",&xx,&yy,&zz);
add(xx,yy,zz),add(yy,xx,zz);
}
Dijkstra();
for(int i=1;i<=n;i++)
for(int j=first[i];~j;j=next[j])
if(rec[v[j]]!=rec[i]){
node[++cnt].from=rec[i],node[cnt].now=rec[v[j]];
node[cnt].weight=dis[i]+w[j]+dis[v[j]];
}
Kruskal(),memset(vis,0,sizeof(vis));
for(int i=1;i<=n;i++)
if(Station[i]&&!vis[i])
T++,dfs(i,-1);
scanf("%d",&q);
for(int i=1;i<=q;i++){
scanf("%d%d%d",&xx,&yy,&zz);
if(vis[xx]==vis[yy]&&lca(xx,yy)<=zz)puts("TAK");
else puts("NIE");
}
}
BZOJ 4144 Dijkstra+Kruskal+倍增LCA的更多相关文章
- BZOJ 3732 Network Kruskal+倍增LCA
题目大意:给定一个n个点m条边的无向连通图.k次询问两点之间全部路径中最长边的最小值 NOIP2013 货车运输.差点儿就是原题...仅仅只是最小边最大改成了最大边最小.. . 首先看到最大值最小第一 ...
- BZOJ 3732 Network —— 最小生成树 + 倍增LCA
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3732 Description 给你N个点的无向图 (1 <= N <= 15, ...
- cf827D Best Edge Weight (kruskal+倍增lca+并查集)
先用kruskal处理出一个最小生成树 对于非树边,倍增找出两端点间的最大边权-1就是答案 对于树边,如果它能被替代,就要有一条非树边,两端点在树上的路径覆盖了这条树边,而且边权不大于这条树边 这里可 ...
- 【BZOJ3732】 Network Kruskal+倍增lca
Description 给你N个点的无向图 (1 <= N <= 15,000),记为:1…N. 图中有M条边 (1 <= M <= 30,000) ,第j条边的长度为: d_ ...
- cf609E Minimum Spanning Tree For Each Edge (kruskal+倍增Lca)
先kruskal求出一个最小生成树,然后对于每条非树边(a,b),从树上找a到b路径上最大的边,来把它替换掉,就是包含这条边的最小生成树 #include<bits/stdc++.h> # ...
- BZOJ.4144.[AMPPZ2014]Petrol(Kruskal重构树)
BZOJ 看别人代码的时候发现哪一步都很眼熟,突然想起来,就在四个月前我好像看过还给别人讲过?mmp=v= 果然不写写就是容易忘.写了好歹忘了的时候还能复习呢(虽然和看别人的好像也没多少差别?). 首 ...
- BFS+最小生成树+倍增+LCA【bzoj】4242 水壶
[bzoj4242 水壶] Description JOI君所居住的IOI市以一年四季都十分炎热著称. IOI市是一个被分成纵H*横W块区域的长方形,每个区域都是建筑物.原野.墙壁之一.建筑物的区域有 ...
- BZOJ 2243: [SDOI2011]染色 树链剖分 倍增lca 线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- [BZOJ 4144] Petrol
Link: BZOJ 4144 传送门 Solution: 一道不错的图论综合题 因为只询问关键点,因此重点是要求出关键点之间的最短路,以最短路建图 记$nst[i]$为离$i$最近的关键点:可以发现 ...
随机推荐
- 紫书 例题 11-14 UVa 1279 (动点最小生成树)(详细解释)
这道题写了好久-- 在三维空间里面有动的点, 然后求有几次最小生成树. 其实很容易发现, 在最小生成树切换的时候,在这个时候一定有两条边相等, 而且等一下更大的那条边在最小生成树中,等一下更小的边不在 ...
- 【BZOJ 1193】 [HNOI2006]马步距离
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 原问题可以等价为两个点. 然后其中一个点要移动到另外一个点. 那么我们可以把左下角那个点(对称总是可以得到一个点在左下角)放在原点的 ...
- 洛谷 P3671 [USACO17OPEN]Where's Bessie? 贝西在哪呢
P3671 [USACO17OPEN]Where's Bessie? 贝西在哪呢 题目背景 农夫John正在测试一个他新发明的全自动寻找奶牛无人机,它能够照一张农场的图片然后自动找出奶牛的位置. 不幸 ...
- 安装spark问题汇总
使用的版本是 spark-1.6.3-bin-without-hadoop 运行spark-shell报错 运行spark-sql报错找不到org.datanucleus.api.jdo.JDOPer ...
- css3模糊图片
image的css样式 filter: blur(10px); -webkit-filter: blur(10px); -moz-filter: blur(10px); -ms-filter: blu ...
- thinkphp5的Illegal string offset 'id'错误
thinkphp5的Illegal string offset 'id'错误 问题 解答 数组同名了,一个html页面传进来两个cateres的数组,所以在找id的时候不知道找这两个里面的哪一个 第一 ...
- es6 --- 功能
标记的模板文字 1.模板文字!确实很棒.我们不再会这样做…. const concatenatedString = "I am the " + number + "per ...
- 网络流Dinic算法模板 POJ1273
这就是以后我的板子啦~~~ #include <queue> #include <cstdio> #include <cstring> #include <a ...
- POJ 2449 第k短路 Dijkstra+A*
这道题我拖了半年,,,终于写出来了 思路: 先反向建边 从终点做一次最短路 ->这是估价函数h(x) 再正常建边,从起点搜一遍 (priority_queue(h(x)+g(x))) g(x)是 ...
- 基于python3-sklearn,Flask 的回归预测系统
看到一副图片挺有意思,放在片头 序 "傍晚小街路面上沁出微雨后的湿润,和煦的西风吹来,抬头看看天边的晚霞,嗯明天又是一个好天气.走到水果摊旁,挑了个根蒂蜷缩.敲起来声音浊响的青绿西瓜,一边满 ...