【BZOJ 3669】 [Noi2014]魔法森林 LCT维护动态最小生成树
这道题看题意是在求一个二维最小瓶颈路,唯一可行方案就是枚举一维在这一维满足的条件下使另一维最小,那么我们就把第一维排序利用A小的边在A大的情况下仍成立来动态加边维护最小生成树。
#include <cstdio>
#include <algorithm>
namespace Pre{
inline void read(int &sum){
register char ch=getchar();
for(sum=;ch<''||ch>'';ch=getchar());
for(;ch>=''&&ch<='';sum=(sum<<)+(sum<<)+ch-'',ch=getchar());
}
const int N=;
const int M=;
const int Inf=0x7f7f7f7f;
int A[M],B[M],ans=Inf,X[M],Y[M],n,m,e[M];
inline bool comp(int a,int b){
return A[a]<A[b];
}
inline int Max(int x,int y){
return x>y?x:y;
}
inline int Min(int x,int y){
return x<y?x:y;
}
}
namespace LCT{
const int MN=;
struct Node{
Node *ch[],*f;
int id,rid;
bool rev;
inline void pushup(){
using Pre::B;
id=B[ch[]->id]>B[ch[]->id]?ch[]->id:ch[]->id;
id=B[id]>B[rid]?id:rid;
}
inline void pushdown();
}null[MN];
inline void Swap(Node *&a,Node *&b){
Node *c=a;
a=b;
b=c;
}
inline void Node:: pushdown(){
if(!rev)return;
Swap(ch[],ch[]);
ch[]->rev^=;
ch[]->rev^=;
rev=;
}
inline void Init(){
using namespace Pre;
null->ch[]=null->ch[]=null->f=null;
for(int i=;i<=n;i++)
null[i].ch[]=null[i].ch[]=null[i].f=null,null[i].id=null[i].rid=;
for(int i=n+;i<=n+m;i++)
null[i].ch[]=null[i].ch[]=null[i].f=null,null[i].id=null[i].rid=i-n;
}
inline int get(Node *p){
return p->f->ch[]==p;
}
inline bool isroot(Node *p){
return p->f->ch[]!=p&&p->f->ch[]!=p;
}
inline void rotate(Node *p){
Node *fa=p->f,*pa=fa->f;
int j=get(p);
if(!isroot(fa))pa->ch[get(fa)]=p;
if((fa->ch[j]=p->ch[j^])!=null)fa->ch[j]->f=fa;
fa->f=p;
p->ch[j^]=fa;
p->f=pa;
fa->pushup();
p->pushup();
}
inline void spaly(Node *p){
p->pushdown();
for(Node *fa=p->f;!isroot(p);rotate(p),fa=p->f)
if(!isroot(fa)){
fa->f->pushdown(),fa->pushdown(),p->pushdown();
rotate(get(fa)==get(p)?fa:p);
}else
fa->pushdown(),p->pushdown();
}
inline void expose(Node *p){
Node *y=null;
while(p!=null){
spaly(p);
p->ch[]=y;
p->pushup();
y=p;
p=p->f;
}
}
inline void make_root(Node *p){
expose(p);
spaly(p);
p->rev^=;
}
inline Node *find_root(Node *p){
expose(p);
spaly(p);
while(p->ch[]!=null)
p=p->ch[],p->pushdown();
return p;
}
inline void link(Node *a,Node *b){
make_root(a);
a->f=b;
}
inline void cut(Node *a,Node *b){
make_root(a);
expose(b);
spaly(b);
b->ch[]->f=null;
b->ch[]=null;
b->pushup();
}
inline void Link(int a,int b){
link(null+a,null+b);
}
inline void Cut(int a,int b){
cut(null+a,null+b);
}
inline int find(int x){
return find_root(null+x)-null;
}
inline int query(int a,int b){
Node *x=null+a,*y=null+b;
make_root(x);
expose(y);
spaly(y);
return Pre::B[y->id];
}
inline int query_id(int a,int b){
Node *x=null+a,*y=null+b;
make_root(x);
expose(y);
spaly(y);
return y->id;
}
}
inline void Init(){
using namespace Pre;
read(n),read(m);
for(int i=;i<=m;i++)
read(X[i]),read(Y[i]),read(A[i]),read(B[i]),e[i]=i;
std::sort(e+,e+m+,comp);
LCT::Init();
}
inline void Work(){
using namespace Pre;
using LCT::find;
using LCT::query;
using LCT::Link;
using LCT::Cut;
using LCT::query_id;
for(int i=;i<=m;i++)
if(find(X[e[i]])!=find(Y[e[i]])){
Link(e[i]+n,X[e[i]]),Link(e[i]+n,Y[e[i]]);
if(find()==find(n))
ans=Min(query(,n)+A[e[i]],ans);
}else if(query(X[e[i]],Y[e[i]])>B[e[i]]){
int temp=query_id(X[e[i]],Y[e[i]]);
Cut(temp+n,X[temp]),Cut(temp+n,Y[temp]);
Link(e[i]+n,X[e[i]]),Link(e[i]+n,Y[e[i]]);
if(find()==find(n))
ans=Min(ans,query(,n)+A[e[i]]);
}
}
int main(){
using namespace Pre;
Init(),Work();
printf("%d",ans==Inf?-:ans);
return ;
}
【BZOJ 3669】 [Noi2014]魔法森林 LCT维护动态最小生成树的更多相关文章
- BZOJ 3669: [Noi2014]魔法森林( LCT )
排序搞掉一维, 然后就用LCT维护加边MST. O(NlogN) ------------------------------------------------------------------- ...
- bzoj 3669: [Noi2014]魔法森林 (LCT)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=3669 题面: 3669: [Noi2014]魔法森林 Time Limit: 30 Sec ...
- bzoj 3669: [Noi2014] 魔法森林 LCT版
Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节 ...
- BZOJ 3669: [Noi2014]魔法森林 [LCT Kruskal | SPFA]
题目描述 为了得到书法大家的真传,小 E 同学下定决心去拜访住在魔法森林中的隐 士.魔法森林可以被看成一个包含 n 个节点 m 条边的无向图,节点标号为 1,2,3,…,n,边标号为 1,2,3,…, ...
- BZOJ 3669: [Noi2014]魔法森林(lct+最小生成树)
传送门 解题思路 \(lct\)维护最小生成树.我们首先按照\(a\)排序,然后每次加入一条边,在图中维护一棵最小生成树.用并查集判断一下\(1\)与\(n\)是否联通,如果联通的话就尝试更新答案. ...
- bzoj 3669: [Noi2014]魔法森林
bzoj 3669: [Noi2014]魔法森林 Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号 ...
- bzoj 3669: [Noi2014]魔法森林 动态树
3669: [Noi2014]魔法森林 Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 363 Solved: 202[Submit][Status] ...
- bzoj 3669: [Noi2014]魔法森林 -- 动点spfa
3669: [Noi2014]魔法森林 Time Limit: 30 Sec Memory Limit: 512 MB 动点spfa Description 为了得到书法大家的真传,小E同学下定决心 ...
- [BZOJ 3669] [Noi2014] 魔法森林 【LCT】
题目链接:BZOJ - 3669 题目分析 如果确定了带 x 只精灵A,那么我们就是要找一条 1 到 n 的路径,满足只经过 Ai <= x 的边,而且要使经过的边中最大的 Bi 尽量小. 其实 ...
随机推荐
- 阻塞队列之LinkedBlockingQueue
概述 LinkedBlockingQueue内部由单链表实现,只能从head取元素,从tail添加元素.添加元素和获取元素都有独立的锁,也就是说LinkedBlockingQueue是读写分离的,读写 ...
- python-集合类型
集合具有唯一性(集合中的元素各不相同),无序性,确定性(集合中的元素是不可改变的,不能是列表,字典以及集合本身) 1.add(self, *args, **kwargs),union(self, *a ...
- 实现一个带有指纹加密功能的笔记本(Android)第一部分
自己经常会忘记一些密码什么的,想把这些密码保存下来,但是别人做的软件总有一点不安全的感觉,所以自己动手做了一个带有指纹加密的笔记本. 以下是本工程用到的一些第三方包 compile 'org.gree ...
- .Net 面试题 汇总(二)
51..net中读写XML的类都归属于哪些命名空间? 答:System.Xml 52.解释一下UDDI.WSDL的意义及其作用. 答:UDDI即统一描述.发现和集成协议.作用: 用来说明一个Web服务 ...
- VS2017 远程调试小记
VS2017 远程调试小记 支持windows\linux\macos, 直接连接项目点的上线版本代码进行调试.保证bug在同个环境下实时追踪. 注意点 双方的 msvsmon.exe版本需一致,最好 ...
- stm32--USB(作为U盘)+FatFs的实现
一.USB功能的添加(作为U盘) 添加文件 将官方库中的Library文件夹中的所有有效文件添加到工程中,分为4个文件夹: usb class为硬件相关(Library\Class): usb dri ...
- NodeJS微信公众平台开发
微信是手机用户必备的App,微信最开始只是作为社交通讯应用供用户使用,但随着用户量不断的增加,微信的公众号在微信上表现出来了它强大的一面,微信公众平台具有四大优势:1.平台更加稳固:2.用户关系更加平 ...
- 使用polarssl进行RSA加密解密
RSA算法的原理就不提了,网上有很多介绍的文章,因为项目中使用RSA加密,所以需要找一个RSA加密的算法,之前尝试过使用Crypto++库,无奈Crypto++其中使用了大量的模版,各种继承,看着头大 ...
- [转][赞]Android开发者必知的开发资源
英文原文:Bongzimo 翻译: ImportNew-黄小非 随着Android平台市场份额的持续猛增 ,越来越多的开发者开始投入Android应用程序的开发大潮.如果您是一位2013年刚刚入行的 ...
- 「日常训练」The Intriguing Obsession(CodeForces Round #439 Div.2 C)
2018年11月30日更新,补充了一些思考. 题意(CodeForces 869C) 三堆点,每堆一种颜色:连接的要求是同色不能相邻或距离必须至少3.问对整个图有几种连接方法,对一个数取模. 解析 要 ...