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 ...
随机推荐
- C# IsAssignableFrom与IsSubClassOf 判断匿名类是否继承父类
public class Dog : Animal { public string name { get; set; } } public class Animal { public string i ...
- windebug常用命令
使用~查看所有线程 切换到一号线程:~1s 查看所有线程的托管堆栈 ~* e!clrstack 怎么查看,当前线程下,变量的信息? 对于托管代码而言,最核心的命令就是!do(dump object的 ...
- ORA-00906 missing left parenthesis括号
Oracle 建表报错:ORA-00906 missing left parenthesis括号 建表语句:create table test(id char,name varchar(1),s ...
- ActiveMQ监听消息并进行转发,监听不同的mq服务器和不同的队列
工作中刚接触mq消息业务,其实也就是监听一下别的项目发送的消息然后进行对应的转发,但是监听的mq会有多个,而且转发的地址也可能有多个,这里就使用spring集成的方式!记录一下实现方式: 监听多个mq ...
- 使用IntelliJ IDEA新建maven的javaWeb项目部署,启动访问index,jsp页面
对于用惯了eclipse的人,idea其实还挺不一样的,也是摸索了很久,看了好多博客,这里就记录一下,以后肯定经常用!,不过使用熟练了,功能确实非常强大,真的牛! 1 新建maven项目,配置好目录结 ...
- python函数,模块及eclipse配置python开发环境
一.eclipse的使用 1.作用 (1)最好用的IDE (2)可调式debug (3)查看可执行过程 (4)可查看源代码 2.安装eclipse及配置 目录安装Pythonpython for ec ...
- 跳出python的各种坑(1)
2017-11-1915:38:17 一定要跳出python的各种坑,一开始遇到的好多思维上的认知错误,因为刚开始学习,对python是个什么都不清楚,所以记录一下自己遇到的各种坑.不用担心自己遇到的 ...
- spin_lock & mutex_lock的区别? 【转】
转自:http://blog.csdn.net/wilsonboliu/article/details/19190861 本文由该问题引入到内核锁的讨论,归纳如下 为什么需要内核锁? 多核处理器下 ...
- LVS ARP广播产生的问题和处理方式【转】
转自 LVS ARP广播产生的问题和处理方式-htckiller2010-ChinaUnix博客http://blog.chinaunix.net/uid-24960107-id-193084.htm ...
- python3 asyncio官方文档中文版
事件循环基类 事件循环基类 事件循环是由asyncio提供的核心执行装置.它提供了多种服务,包括: 注册.执行和关闭延时调用(超时) 为各种通信创建客户端和服务端传输 为一个外部程序通信启动子进程和相 ...