【洛谷4172】 [WC2006]水管局长(LCT)
传送门
Solution
如果不需要动态的话,那就是一个裸的最小生成树上的最大边权对吧。
现在动态了的话,把这个过程反着来,就是加边对吧。
现在问题变成了怎么动态维护加边的最小生成树,这是一个比较常见的套路了,使用LCT(暴力)。。。
深刻理解一下Kruscal的过程,就是每一次选不在连通块内的最小边权。
那么我们如果加了一条边,显然会形成一个环,我们只要删除这个环里面最大的边权就好了。
所以LCT动态维护路径上的最大值然后根据情况\(Cut*2+Link*2\)就好了。。。
代码实现
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<queue>
#include<set>
#include<map>
#include<iostream>
using namespace std;
#define re register
#define ll long long
inline int gi()
{
int f=1,sum=0;char ch=getchar();
while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0' && ch<='9'){sum=(sum<<3)+(sum<<1)+ch-'0';ch=getchar();}
return f*sum;
}
typedef pair<int,int> pii;
#define mp make_pair
const int N=100010,Mm=1000010;
struct edge{int u,v,w;bool operator<(const edge &b)const{return w<b.w;}}e[Mm];
struct ques{int opt,x,y;}q[Mm];
struct node{int ff,ch[2],rev,mx,val;}t[(N+Mm)<<1];
map<pii,int>M;
int n,m,Q,Cut[Mm],ans[Mm];
void init(int id,int val){t[id].ch[0]=t[id].ch[1]=0;t[id].rev=0;t[id].ff=0;t[id].val=t[id].mx=val;}
namespace LinkCutTree
{
int sta[(N+Mm)<<1],top;
void pushup(int x)
{
t[x].mx=t[x].val;
if(e[t[x].mx].w<e[t[t[x].ch[0]].mx].w)t[x].mx=t[t[x].ch[0]].mx;
if(e[t[x].mx].w<e[t[t[x].ch[1]].mx].w)t[x].mx=t[t[x].ch[1]].mx;
}
void reverse(int x){t[x].rev^=1;swap(t[x].ch[0],t[x].ch[1]);}
void pushdown(int x){if(!t[x].rev)return;if(t[x].ch[0])reverse(t[x].ch[0]);if(t[x].ch[1])reverse(t[x].ch[1]);t[x].rev^=1;}
bool isroot(int x){return (t[t[x].ff].ch[0]!=x)&(t[t[x].ff].ch[1]!=x);}
void rotate(int x)
{
int y=t[x].ff,z=t[y].ff;
if(!isroot(y))t[z].ch[t[z].ch[1]==y]=x;
t[x].ff=z;int k=t[y].ch[1]==x;
t[y].ch[k]=t[x].ch[k^1];t[t[x].ch[k^1]].ff=y;
t[x].ch[k^1]=y;t[y].ff=x;
pushup(y);
}
void splay(int x)
{
sta[++top]=x;
for(int pos=x;!isroot(pos);pos=t[pos].ff)sta[++top]=t[pos].ff;
while(top)pushdown(sta[top--]);
while(!isroot(x))
{
int y=t[x].ff,z=t[y].ff;
if(!isroot(y))
(t[y].ch[1]==x)^(t[z].ch[1]==y)?rotate(x):rotate(y);
rotate(x);
}
pushup(x);
}
void access(int x){for(int y=0;x;y=x,x=t[x].ff)splay(x),t[x].ch[1]=y,pushup(x);}
void makeroot(int x){access(x);splay(x);reverse(x);}
int findroot(int x){access(x);splay(x);while(t[x].ch[0])x=t[x].ch[0];return x;}
void split(int x,int y){makeroot(x);access(y);splay(y);}
void link(int x,int y){makeroot(x);t[x].ff=y;}
void cut(int x,int y){split(x,y);t[x].ff=t[y].ch[0]=0;}
}
using namespace LinkCutTree;
int main()
{
n=gi();m=gi();Q=gi();
for(int i=1;i<=m;i++){int u=gi(),v=gi(),w=gi();if(u>v)swap(u,v);e[i]=(edge){u,v,w};}
sort(e+1,e+m+1);
for(int i=1;i<=m;i++)M[mp(e[i].u,e[i].v)]=i;
for(int i=1;i<=n+m;i++)init(i,i<=n?0:i-n);
int tot=0;
for(int i=1;i<=Q;i++){int opt=gi(),x=gi(),y=gi();if(x>y)swap(x,y);q[i]=(ques){opt,x,y};if(q[i].opt==2)Cut[M[mp(q[i].x,q[i].y)]]=1;}
for(int i=1;i<=m;i++)
if(!Cut[i])
{
int u=e[i].u,v=e[i].v;
if(tot==n-1)break;
if(findroot(u)!=findroot(v))
{
link(u,i+n);
link(v,i+n);
tot++;
}
}
for(int i=Q;i>=1;i--)
{
int x=q[i].x,y=q[i].y;
split(x,y);
if(q[i].opt&1){ans[i]=e[t[y].mx].w;}
else
{
int id=M[mp(x,y)],d=t[y].mx;
if(e[d].w<=e[id].w)continue;
cut(e[d].u,d+n);cut(d+n,e[d].v);
link(x,id+n);link(y,id+n);
}
}
for(int i=1;i<=Q;i++)
if(q[i].opt&1)printf("%d\n",ans[i]);
return 0;
}
【洛谷4172】 [WC2006]水管局长(LCT)的更多相关文章
- 洛谷.4172.[WC2006]水管局长(LCT Kruskal)
题目链接 洛谷(COGS上也有) 不想去做加强版了..(其实处理一下矩阵就好了) 题意: 有一张图,求一条x->y的路径,使得路径上最长边尽量短并输出它的长度.会有<=5000次删边. 这 ...
- 洛谷4172 WC2006水管局长(LCT维护最小生成树)
这个题和魔法森林感觉有很相近的地方啊 同样也是维护一个类似最大边权最小的生成树 但是不同的是,这个题是有\(cut\)和询问,两种操作.... 这可如何是好啊? 我们不妨倒着来考虑,假设所有要\(cu ...
- 洛谷 4172 [WC2006]水管局长
[题解] 我们把操作倒过来做,就变成了加边而不是删边.于是用LCT维护动态加边的最小生成树就好了.同样要注意把边权变为点权. #include<cstdio> #include<al ...
- 洛谷P4172 [WC2006]水管局长 (LCT,最小生成树)
洛谷题目传送门 思路分析 在一个图中,要求路径上最大边边权最小,就不难想到最小生成树.而题目中有删边的操作,那肯定是要动态维护啦.直接上LCT维护边权最小值(可以参考一下蒟蒻的Blog) 这时候令人头 ...
- 洛谷P4172 [WC2006]水管局长(lct求动态最小生成树)
SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一条从A至B的水管的路径, ...
- [洛谷P4172] WC2006 水管局长
问题描述 SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一条从A至B的水 ...
- P4172 [WC2006]水管局长 LCT维护最小生成树
\(\color{#0066ff}{ 题目描述 }\) SC 省 MY 市有着庞大的地下水管网络,嘟嘟是 MY 市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的 ...
- luogu P4172 [WC2006]水管局长 LCT维护动态MST + 离线
Code: #include<bits/stdc++.h> #define maxn 1200000 #define N 120000 using namespace std; char ...
- 【洛谷P4172】水管局长
题目大意:给定 N 个点,M 条边的无向图,支持两种操作:动态删边和查询任意两点之间路径上边权的最大值最小是多少. 题解: 引理:对原图求最小生成树,可以保证任意两点之间的路径上边权的最大值取得最小值 ...
- P4172 [WC2006]水管局长(LCT)
P4172 [WC2006]水管局长 LCT维护最小生成树,边权化点权.类似 P2387 [NOI2014]魔法森林(LCT) 离线存储询问,倒序处理,删边改加边. #include<iostr ...
随机推荐
- WebService关于Unable to find required classes (javax.activation.DataHandler and javax.mail.internet.MimeMultipart)问题解决
错误原因:需要mail.jar和activation.jar. Solution:Web Services Required Jars Download Instructions http://www ...
- python中的深浅copy
https://www.cnblogs.com/Eva-J/p/5534037.html 转自Eva_J 分析的特别好
- Mysql 数据库管理
1. mysql用户和权限 mysql权限控制通过两步控制,第一是能不能连接(验证用户身份),第二是能执行什么操作(验证权限). 其中身份验证,需要验证连接mysql的计算机的IP地址或者计算机名称还 ...
- 图解Go select语句原理
Go 的select语句是一种仅能用于channl发送和接收消息的专用语句,此语句运行期间是阻塞的:当select中没有case语句的时候,会阻塞当前的groutine.所以,有人也会说select是 ...
- 第二阶段第十次spring会议
昨天我对宠物信息进行了添加和调整. 今天我将继续对宠物信息添加保存按钮. private void button2_Click(object sender, EventArgs e) { Settin ...
- oracle 12c centos 7 安装配置
1,安装centos 7 安装 宿主机配置信息: 内存:8G 系统盘:30G swap分区:30G (笑了oracle安装自检不过) /u01(50G) :oracle安装目录 /u02(50G): ...
- ----改写superheros的json以及上传到github----
以下为js代码: var header = document.querySelector('header'); var section = document.querySelector('sectio ...
- mysql 主从库同步
#主库修改my.ini [mysqld] server log-bin=mysql-bin binlog-do-db=demo #从库修改my.ini [mysqld] server replicat ...
- Linux学习--gdb调试
一.gdb常用命令: 命令 描述 backtrace(或bt) 查看各级函数调用及参数 finish 连续运行到当前函数返回为止,然后停下来等待命令 frame(或f) 帧编号 选择栈帧 info(或 ...
- 解决在静态页面上使用动态参数,造成spider多次和重复抓取的问题
我们在使用百度统计中的SEO建议检查网站时,总是发现“静态页参数”一项被扣了18分,扣分原因是“在静态页面上使用动态参数,会造成spider多次和重复抓取”.一般来说静态页面上使用少量的动态参数的话并 ...