4144: [AMPPZ2014]Petrol (多源最短路+最小生成树+启发式合并)
4144: [AMPPZ2014]Petrol
Time Limit: 10 Sec Memory Limit: 256 MB
Submit: 752 Solved: 298
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
1 5 2 6
1 3 1
2 3 2
3 4 3
4 5 5
6 4 5
4
1 2 4
2 6 9
1 5 9
6 5 8
Sample Output
TAK
TAK
NIE
HINT
Source
朴素是先写个多源最短路把加油站的最小生成树所有可能边弄出来,然后后面按部就班地建个最小生成树,然后对树上写个主席树(st倍增)找u到v上最大值,和b比较一下。
那我们在把所有可能边弄出来以后停一下2333,我们连边不要对这条边俩端点连,而是对他们所在块的堆头节点连,这个建出来在查询最大值上是等效的。然后我们启发式地合并堆,这样可以把堆高度控制在logn,就不用再写个主席树啥的那麻烦了。直接两个端点往上一个一个跳,logn的不会有事的。
#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;
typedef pair<LL,int > pli;
const int N = 2e5+;
bool need[N];
vector<pli> e[N];
struct node
{
int u,v;
LL w;
node(int _u=,int _v=,LL _w=):u(_u),v(_v),w(_w) {}
};
vector<node> ve;
int n,s,m,T;
priority_queue<pli,vector<pli>,greater<pli> > que;
int fuel[N];
int fa[N],rfa[N],rk[N],dep[N];
LL dis[N],val[N];
bool vis[N];
int pre[N];
void dij()
{
while(!que.empty())
{
pli now=que.top();
que.pop();
LL dist=now.fi;
int u=now.se;
if(vis[u]) continue;
vis[u]=;
int sz=e[u].size();
for(int i=;i<sz;i++)
{
pli p=e[u][i];
int w=p.fi;
int v=p.se;
if(!pre[v] || dis[v]>dis[u]+w)
{
dis[v]=dis[u]+w;
pre[v]=pre[u];
que.push(mp(dis[v],v));
}
else if(pre[u] != pre[v])
ve.pb(node(pre[u],pre[v],dis[u]+dis[v]+w));
}
}
return ;
}
bool cmp(node a,node b)
{
return a.w<b.w;
}
int Find(int u)
{
if(fa[u]!=u)
fa[u]=Find(fa[u]);
return fa[u];
}
void Union()
{
sort(ve.begin(),ve.end(),cmp);
for(int i=;i<=s;i++)
{
fa[fuel[i]]=fuel[i];
rk[fuel[i]]=;
}
int sz=ve.size();
for(int i=;i<sz;i++)
{
node p=ve[i];
int u = p.u, v = p.v;
LL w = p.w;
u = Find(u), v = Find(v);
if(u==v) continue;
if(rk[u]<rk[v]) swap(u,v);
if(rk[u]==rk[v]) rk[u]++;
rfa[v]=u,fa[v]=u,val[v]=w;
}
return ;
}
void dealdep(int u)
{
if(dep[u]>) return ;
if(fa[u]==u)
{
dep[u]=;
return ;
}
dealdep(rfa[u]);
dep[u]=dep[rfa[u]]+;
return ;
}
bool solve(int u,int v,LL b)
{
if(Find(u)!=Find(v)) return ;
if(dep[u]<dep[v]) swap(u,v);
while(dep[u]>dep[v])
{
if(b<val[u]) return ;
u=rfa[u];
}
if(u==v) return ;
while(u!=v)
{
if(b<val[u]) return ;
if(b<val[v]) return ;
u=rfa[u];
v=rfa[v];
}
return ;
}
int main()
{
scanf("%d%d%d",&n,&s,&m);
clr_1(dis);
for(int i=;i<=s;i++)
{
scanf("%d",fuel+i);
que.push(mp(,fuel[i]));
pre[fuel[i]]=fuel[i];
dis[fuel[i]]=;
}
for(int i=;i<=m;i++)
{
int u,v;
LL w;
scanf("%d%d%lld",&u,&v,&w);
e[u].pb(mp(w,v));
e[v].pb(mp(w,u));
}
dij();
Union();
for(int i=;i<=s;i++)
dealdep(fuel[i]);
int u,v;
LL b,maxn;
int q;
scanf("%d",&q);
while(q--)
{
scanf("%d%d%lld",&u,&v,&b);
if(solve(u,v,b))
printf("TAK\n");
else
printf("NIE\n");
}
return ;
}
4144: [AMPPZ2014]Petrol (多源最短路+最小生成树+启发式合并)的更多相关文章
- bzoj 4242 水壶 (多源最短路+最小生成树+启发式合并)
4242: 水壶 Time Limit: 50 Sec Memory Limit: 512 MBSubmit: 1028 Solved: 261[Submit][Status][Discuss] ...
- [BZOJ4144][AMPPZ2014]Petrol[多源最短路+MST]
题意 题目链接 分析 假设在 \(a \rightarrow b\) 的最短路径中出现了一个点 \(x\) 满足到 \(x\) 最近的点是 \(c\) ,那么我们完全可以从 \(a\) 直接走到 \( ...
- BZOJ 4144: [AMPPZ2014]Petrol
4144: [AMPPZ2014]Petrol Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 457 Solved: 170[Submit][Sta ...
- BZOJ.4144.[AMPPZ2014]Petrol(Kruskal重构树)
BZOJ 看别人代码的时候发现哪一步都很眼熟,突然想起来,就在四个月前我好像看过还给别人讲过?mmp=v= 果然不写写就是容易忘.写了好歹忘了的时候还能复习呢(虽然和看别人的好像也没多少差别?). 首 ...
- 【BZOJ】4144: [AMPPZ2014]Petrol
题意 给定一个\(n\)个点.\(m\)条边的带权无向图,其中有\(s\)个点是加油站.每辆车都有一个油量上限\(b\),即每次行走距离不能超过\(b\),但在加油站可以补满.\(q\)次询问,每次给 ...
- 【BZOJ4144】[AMPPZ2014]Petrol 最短路+离线+最小生成树
[BZOJ4144][AMPPZ2014]Petrol Description 给定一个n个点.m条边的带权无向图,其中有s个点是加油站. 每辆车都有一个油量上限b,即每次行走距离不能超过b,但在加油 ...
- 【BZOJ4144】[AMPPZ2014]Petrol(最短路+最小生成树+并查集)
Description 给定一个n个点.m条边的带权无向图,其中有s个点是加油站. 每辆车都有一个油量上限b,即每次行走距离不能超过b,但在加油站可以补满. q次询问,每次给出x,y,b,表示出发点是 ...
- [AMPPZ2014]Petrol
关键点的最小生成树? 关键点初始化为0,跑多源最短路,然后重构整个图,用Kruskal跑最小生成树 然后跑树链剖分在线回答询问 对树上每个点维护到链顶的最大值,结合线段树可以做到\(\Theta(n ...
- Day3 最短路 最小生成树 拓扑排序
Day3 最短路 最小生成树 拓扑排序 (一)最短路 一.多源最短路 从任意点出发到任意点的最短路 1. Floyd \(O(n^3)\) for(int k=1;k<=n;k++) for(i ...
随机推荐
- 关于UNIX的exec函数
在UNIX系统中,系统为进程相关提供了一系列的控制原语,包括:进程fork,进程exit,进程exec,进程wait等服务. 该篇文章主要与进程exec服务有关,并记录了几个需要注意留意的点. 照例给 ...
- sqlplus设置长度
1.set linesize 100 2.col XX format a30 3.col XXX format 9,999,999,999 3.set heading off 表头不显示
- 关于new Handler()与new Handler(Looper.getMainLooper())区别
如果你不带参数的实例化:Handler handler=new Handler();那么这个会默认用当前线程的Looper对象. 一般而言,如果你的Handler是要用来刷新UI的,那么就需要在主线程 ...
- 利用rundll32执行程序的函数执行程序
1.前言 无意间发现hexacorn这个国外大佬,给出了很多通过rundll32执行DLL中的函数执行程序的方法,思路很灵巧. 2.原理 rundll32加载dll 用法: rundll32 < ...
- C#事件实现文件下载时进度提醒
C#中的事件是建立在委托的基础上,标准的事件模型应该包括以下几点: 声明一个用于定义事件的委托,这里用系统自带的泛型委托原型EventHandler<TEventArgs>,如:publi ...
- Shell-输入密码转换为*
Code: read -p "请输入使用者都名称:" USER echo -e "请输入使用者密码: \c" while : ;do char=` #这里是反引 ...
- jQuery简单介绍
一.jQuery介绍 jQuery是一个轻量级的.兼容多浏览器的JavaScript库. jQuery使用户能够更方便地处理HTML Document.Events.实现动画效果.方便地进行Ajax交 ...
- Webmin试玩
安装: # cd /opt # wget http://www.webmin.com/jcameron-key.asc # wget http://www.webmin.com/download/rp ...
- 数据库-mysql数据连接
一:Mysql 连接的使用 在前几章节中,我们已经学会了如果在一张表中读取数据,这是相对简单的,但是在真正的应用中经常需要从多个数据表中读取数据. 本章节我们将向大家介绍如何使用 MySQL 的 JO ...
- Oracle 函数 “数据控制,指定某些人只能查看他权限范围内的信息”
create or replace function work_plan_mask (p_schema VARCHAR2,p_table VARCHAR2) return Varchar2 AS -- ...