BZOJ 2594: [Wc2006]水管局长数据加强版 [LCT kruskal]
2594: [Wc2006]水管局长数据加强版
Time Limit: 25 Sec Memory Limit: 128 MB
Submit: 2917 Solved: 918
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
1 2 2
2 3 3
3 4 2
1 4 2
1 1 4
2 1 4
1 1 4
Sample Output
3
【原题数据范围】
N ≤ 1000
M ≤ 100000
Q ≤ 100000
测试数据中宣布报废的水管不超过5000条;且任何时候我们考虑的水管网络都是连通的,即从任一结点A必有至少一条水管路径通往任一结点B。
【加强版数据范围】
N ≤ 100000
M ≤ 1000000
Q ≤ 100000
任何时候我们考虑的水管网络都是连通的,即从任一结点A必有至少一条水管路径通往任一结点B。
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define pa t[x].fa
#define lc t[x].ch[0]
#define rc t[x].ch[1]
const int N=15e5+,NN=1e5+;
typedef long long ll;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
int n,m,Q; struct node{
int ch[],fa,rev,mx,mxp,w;
}t[N];
inline int wh(int x){return t[pa].ch[]==x;}
inline int isRoot(int x){return t[pa].ch[]!=x&&t[pa].ch[]!=x;}
inline void update(int x){
t[x].mx=t[x].w;t[x].mxp=x;
if(t[lc].mx>t[x].mx) t[x].mx=t[lc].mx,t[x].mxp=t[lc].mxp;
if(t[rc].mx>t[x].mx) t[x].mx=t[rc].mx,t[x].mxp=t[rc].mxp;
}
inline void rever(int x){
t[x].rev^=;
swap(lc,rc);
}
inline void pushDown(int x){
if(t[x].rev){
rever(lc);
rever(rc);
t[x].rev=;
}
}
inline void rotate(int x){
int f=t[x].fa,g=t[f].fa,c=wh(x);
if(!isRoot(f)) t[g].ch[wh(f)]=x;t[x].fa=g;
t[f].ch[c]=t[x].ch[c^];t[t[f].ch[c]].fa=f;
t[x].ch[c^]=f;t[f].fa=x;
update(f);update(x);
}
int st[N],top;
inline void splay(int x){
top=;st[++top]=x;
for(int i=x;!isRoot(i);i=t[i].fa) st[++top]=t[i].fa;
for(int i=top;i>=;i--) pushDown(st[i]); for(;!isRoot(x);rotate(x))
if(!isRoot(pa)) rotate(wh(x)==wh(pa)?pa:x);
} inline void Access(int x){
for(int y=;x;y=x,x=pa){
splay(x);
rc=y;
update(x);
}
}
inline void MakeR(int x){
Access(x);splay(x);
rever(x);
}
inline int FindR(int x){
Access(x);splay(x);
while(lc) x=lc;
return x;
}
inline void Link(int x,int y){
MakeR(x);
t[x].fa=y;
}
inline void Cut(int x,int y){
MakeR(x);Access(y);splay(y);
t[y].ch[]=t[x].fa=;
update(y);//!!!
}
inline void Split(int x,int y){
MakeR(x);Access(y);splay(y);
} int del[N];
struct edge{
int u,v,w,id;
bool operator <(const edge &e)const{return w<e.w;}
}e[N];
inline bool cmpuv(edge a,edge b){
return a.u<b.u||(a.u==b.u&&a.v<b.v);
}
inline bool cmpid(edge a,edge b){
return a.id<b.id;
}
int bs(int u,int v){
int l=,r=m;
while(l<=r){
int mid=(l+r)>>;
if(e[mid].u==u&&e[mid].v==v) return mid;
else if(e[mid].u<u||(e[mid].u==u&&e[mid].v<v)) l=mid+;
else r=mid-;
}
return ;
}
struct ques{
int op,x,y,del,ans;
}q[NN];
int fa[N];
inline int find(int x){return x==fa[x]?x:fa[x]=find(fa[x]);}
void solve(){
for(int i=;i<=n;i++) fa[i]=i;
sort(e+,e++m,cmpid);
int tot=;
for(int i=;i<=m;i++) if(!del[i]){
int x=find(e[i].u),y=find(e[i].v);//printf("hieid %d %d %d %d\n",i,e[i].id,x,y);
if(x!=y){
Link(e[i].u,i+n);Link(e[i].v,i+n);
fa[x]=y;
tot++;
if(tot==n-) break;
}
} for(int i=Q;i>=;i--){
int x=q[i].x,y=q[i].y;
if(q[i].op==){
Split(x,y);
q[i].ans=t[y].mx;//printf("ans %d %d\n",i,t[y].mx);
}else{
int k=q[i].del;//try to add e[k]
Split(x,y);
if(e[k].w<t[y].mx){
int p=t[y].mxp;//printf("replace %d %d\n",p-n,k);
Cut(e[p-n].u,p);Cut(e[p-n].v,p);
Link(x,k+n);Link(y,k+n);
}
}
}
for(int i=;i<=Q;i++) if(q[i].op==) printf("%d\n",q[i].ans);
}
int main(){
//freopen("in.txt","r",stdin);
n=read();m=read();Q=read();
for(int i=;i<=m;i++){
e[i].u=read(),e[i].v=read(),e[i].w=read();
if(e[i].u>e[i].v) swap(e[i].u,e[i].v);
}
sort(e+,e++m);
for(int i=;i<=m;i++){//printf("e %d %d %d %d\n",i,e[i].u,e[i].v,e[i].w);
e[i].id=i;
t[n+i].w=t[n+i].mx=e[i].w;
t[n+i].mxp=n+i;
}
sort(e+,e++m,cmpuv);
for(int i=;i<=Q;i++){
q[i].op=read(),q[i].x=read(),q[i].y=read();
if(q[i].x>q[i].y) swap(q[i].x,q[i].y);
if(q[i].op==){
int k=e[bs(q[i].x,q[i].y)].id;
del[k]=;//printf("del %d\n",k);
q[i].del=k;
}
}
solve();
}
LCT直接保存最大编号也可以
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define pa t[x].fa
#define lc t[x].ch[0]
#define rc t[x].ch[1]
const int N=15e5+,NN=1e5+;
typedef long long ll;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
int n,m,Q; struct node{
int ch[],fa,rev,mxp;
}t[N];
int val[N];
inline int wh(int x){return t[pa].ch[]==x;}
inline int isRoot(int x){return t[pa].ch[]!=x&&t[pa].ch[]!=x;}
inline void update(int x){
t[x].mxp=x;
if(val[t[lc].mxp]>val[t[x].mxp]) t[x].mxp=t[lc].mxp;
if(val[t[rc].mxp]>val[t[x].mxp]) t[x].mxp=t[rc].mxp;
}
inline void rever(int x){
t[x].rev^=;
swap(lc,rc);
}
inline void pushDown(int x){
if(t[x].rev){
rever(lc);
rever(rc);
t[x].rev=;
}
}
inline void rotate(int x){
int f=t[x].fa,g=t[f].fa,c=wh(x);
if(!isRoot(f)) t[g].ch[wh(f)]=x;t[x].fa=g;
t[f].ch[c]=t[x].ch[c^];t[t[f].ch[c]].fa=f;
t[x].ch[c^]=f;t[f].fa=x;
update(f);update(x);
}
int st[N],top;
inline void splay(int x){
top=;st[++top]=x;
for(int i=x;!isRoot(i);i=t[i].fa) st[++top]=t[i].fa;
for(int i=top;i>=;i--) pushDown(st[i]); for(;!isRoot(x);rotate(x))
if(!isRoot(pa)) rotate(wh(x)==wh(pa)?pa:x);
} inline void Access(int x){
for(int y=;x;y=x,x=pa){
splay(x);
rc=y;
update(x);
}
}
inline void MakeR(int x){
Access(x);splay(x);
rever(x);
}
inline int FindR(int x){
Access(x);splay(x);
while(lc) x=lc;
return x;
}
inline void Link(int x,int y){
MakeR(x);
t[x].fa=y;
}
inline void Cut(int x,int y){
MakeR(x);Access(y);splay(y);
t[y].ch[]=t[x].fa=;
update(y);//!!!
}
inline void Split(int x,int y){
MakeR(x);Access(y);splay(y);
}
inline int Que(int x,int y){
MakeR(x);Access(y);splay(y);
return t[y].mxp;
} int del[N];
struct edge{
int u,v,w,id;
bool operator <(const edge &e)const{return w<e.w;}
}e[N];
inline bool cmpuv(edge a,edge b){
return a.u<b.u||(a.u==b.u&&a.v<b.v);
}
inline bool cmpid(edge a,edge b){
return a.id<b.id;
}
int bs(int u,int v){
int l=,r=m;
while(l<=r){
int mid=(l+r)>>;
if(e[mid].u==u&&e[mid].v==v) return mid;
else if(e[mid].u<u||(e[mid].u==u&&e[mid].v<v)) l=mid+;
else r=mid-;
}
return ;
}
struct ques{
int op,x,y,del,ans;
}q[NN];
int fa[N];
inline int find(int x){return x==fa[x]?x:fa[x]=find(fa[x]);}
void solve(){
for(int i=;i<=n;i++) fa[i]=i;
sort(e+,e++m,cmpid);
int tot=;
for(int i=;i<=m;i++) if(!del[i]){
int x=find(e[i].u),y=find(e[i].v);//printf("hieid %d %d %d %d\n",i,e[i].id,x,y);
if(x!=y){
Link(e[i].u,i+n);Link(e[i].v,i+n);
fa[x]=y;
tot++;
if(tot==n-) break;
}
} for(int i=Q;i>=;i--){
int x=q[i].x,y=q[i].y;
if(q[i].op==){
int p=Que(x,y);
q[i].ans=val[p];//printf("ans %d %d\n",i,t[y].mx);
}else{
int k=q[i].del;//try to add e[k]
int p=Que(x,y);
if(e[k].w<val[p]){
Cut(e[p-n].u,p);Cut(e[p-n].v,p);
Link(x,k+n);Link(y,k+n);
}
}
}
for(int i=;i<=Q;i++) if(q[i].op==) printf("%d\n",q[i].ans);
}
int main(){
//freopen("in.txt","r",stdin);
n=read();m=read();Q=read();
for(int i=;i<=m;i++){
e[i].u=read(),e[i].v=read(),e[i].w=read();
if(e[i].u>e[i].v) swap(e[i].u,e[i].v);
}
sort(e+,e++m);
for(int i=;i<=m;i++){//printf("e %d %d %d %d\n",i,e[i].u,e[i].v,e[i].w);
e[i].id=i;
val[n+i]=e[i].w;
t[n+i].mxp=n+i;
}
sort(e+,e++m,cmpuv);
for(int i=;i<=Q;i++){
q[i].op=read(),q[i].x=read(),q[i].y=read();
if(q[i].x>q[i].y) swap(q[i].x,q[i].y);
if(q[i].op==){
int k=e[bs(q[i].x,q[i].y)].id;
del[k]=;//printf("del %d\n",k);
q[i].del=k;
}
}
solve();
}
BZOJ 2594: [Wc2006]水管局长数据加强版 [LCT kruskal]的更多相关文章
- BZOJ 2594: [Wc2006]水管局长数据加强版( LCT )
离线然后就是维护加边的动态MST, Link cut tree秒掉..不过我写+调了好久...时间复杂度O(NlogN + MlogM) ------------------------------- ...
- BZOJ 2594: [Wc2006]水管局长数据加强版(kruskal + LCT)
Description SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一 ...
- BZOJ 2594: [Wc2006]水管局长数据加强版 (LCT维护最小生成树)
离线做,把删边转化为加边,那么如果加边的两个点不连通,直接连就行了.如果联通就找他们之间的瓶颈边,判断一下当前边是否更优,如果更优就cut掉瓶颈边,加上当前边. 那怎么维护瓶颈边呢?把边也看做点,向两 ...
- bzoj 2594: [Wc2006]水管局长数据加强版 动态树
2594: [Wc2006]水管局长数据加强版 Time Limit: 25 Sec Memory Limit: 128 MBSubmit: 934 Solved: 291[Submit][Sta ...
- [BZOJ 2594] [Wc2006]水管局长数据加强版 【LCT】
题目链接:BZOJ - 2594 题目分析 这道题如果没有删边的操作,那么就是 NOIP2013 货车运输,求两点之间的一条路径,使得边权最大的边的边权尽量小. 那么,这条路径就是最小生成树上这两点之 ...
- 【刷题】BZOJ 2594 [Wc2006]水管局长数据加强版
Description SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一 ...
- bzoj 2594: [Wc2006]水管局长数据加强版
Description SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一 ...
- bzoj 2594 [Wc2006]水管局长数据加强版(LCT+最小生成树)
[深坑勿入] [给个链接] http://blog.csdn.net/popoqqq/article/details/41348549 #include<cstdio> #include& ...
- BZOJ2594 [Wc2006]水管局长数据加强版 LCT kruskal
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2594 题意概括 N个点的图,M条带权边.(N<=100000,M<=1000000) ...
随机推荐
- 解决CSS垂直居中的几种方法(基于绝对定位,基于视口单位,Flexbox方法)
在CSS中对元素进行水平居中是非常简单的:如果它是一个行内元素,就对它的父元素应用 text-align: center ;如果它是一个块级元素,就对它自身应用 margin: auto.然而如果要对 ...
- HTML表单属性集合
- Unity Object Pool完全体
using System; using System.Collections.Generic; using UnityEngine; using UnityEngine.Events; public ...
- 用NPOI导出Excel,生成下拉列表、以及下拉联动列表(第1篇/共3篇)
最近帅帅的小毛驴遇到一个很奇葩的需求: 导出Excel报表,而且还要带下拉框,更奇葩的是,下拉框还是联动的. 小毛驴一天比较忙,所以这等小事自然由我来为她分忧了.经历了两天,做了几种解决方案,最后完美 ...
- 如何检测浏览器url变化
用户通过“点击触发”,“操作历史”,“直接访问URL”的方式修改当前URL.这三种触发方式会使浏览器做出不同的行为 html5提供了两种方式在页面中操作历史 history.pushState(sta ...
- Vue2反向代理
前一段时间写了一个vue2的小项目,用的是vue-cli脚手架搭建的项目,项目里需要跨域,但又不能使用jsonp,上网查了一下,发现有一个之前没接触过的词语--反向代理. 什么是"反向代 ...
- 立即掌握SSM框架的要诀
ssm框架的总结: 1. 首先是POM.xml 文件的配置,他的作用主要是添加依懒的关系和自动下载相关的包. 2.对jdbc.properties进行配置 ,作用就是连接你的数据库的配置. 3.对接着 ...
- Python3 的注释
单行注释 # 这是一个注释 print("Hello, World!") 多行注释 1:3个单引号 ''' 这是多行注释,用三个单引号 这是多行注释,用三个单引号 这是多行注释,用 ...
- php的底层原理
PHP说简单,但是要精通也不是一件简单的事.我们除了会使用之外,还得知道它底层的工作原理. PHP是一种适用于web开发的动态语言.具体点说,就是一个用C语言实现包含大量组件的软件框架.更狭义点看,可 ...
- openfire服务器+Spark搭建即时聊天系统 & 阿里云的初步探索
晚上出去和洋仔吃了涮肉,喝了点啤酒,不知不觉就聊到了11点,感觉他工作状态还不错,emmm...都要加油吧.虽然没有当时去山西零下二十多度那么夸张,这几天北京的冬夜还是有点小冷的.好了进入正题: 一. ...