[BZOJ 2594] [Wc2006]水管局长数据加强版 【LCT】
题目链接:BZOJ - 2594
题目分析
这道题如果没有删边的操作,那么就是 NOIP2013 货车运输,求两点之间的一条路径,使得边权最大的边的边权尽量小。
那么,这条路径就是最小生成树上这两点之间的路径。
然而现在有了删边操作,我们就需要一直维护当前的最小生成树。
删边然后维护 MST 还是不会做的,但是加边维护 MST 就可以用 LCT 来做了。于是,我们将询问和操作都记录下来,离线倒着做,就变成加边了。
加边维护 MST 的做法:
对于新加的一条边 (u, v, w) ,我们先求出现有 MST 中 u 到 v 的路径中,边权最大的边,如果这条边权最大的边的边权大于 w ,我们就将这条边删掉,将新加的边连上,加入 MST 。
否则,我们就忽略新加的这条边。
怎样处理边呢?我们把边看做和两个端点分别相连的一个点,即如果有一条边 (u, v) ,标号为 i ,那么我们就是连边 u -> i -> v ,就可以用 LCT 做了。
技巧:Splay 中一个节点就维护它的子树中边权最大的边的标号就可以了。
写代码时出现的错误:新加入一条边 (u, v, w) 时,发现现有 MST u 到 v 的路径上边权最大的边的边权 > w,需要删掉这条边,然后这条边是 T[t] ,t 是提取出的 u 到 v 的路径的 Splay 的根,
于是我 Cut(u, T[t]); Cut(v, T[t]); 然后就...就 0 分了。因为第一个 Cut 做完之后 T[t] 就改变了啊!!需要先记录下来然后再做两次 Cut !
代码
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <cmath> using namespace std; inline void Read(int &Num)
{
char c = getchar();
while (c < '0' || c > '9') c = getchar();
Num = c - '0'; c = getchar();
while (c >= '0' && c <= '9')
{
Num = Num * 10 + c - '0';
c = getchar();
}
} const int MaxN = 100000 + 5, MaxM = 1000000 + 5, MaxT = 1100000 + 5, MaxQ = 100000 + 5; int n, m, q, Top, Tot;
int Father[MaxT], Son[MaxT][2], V[MaxT], T[MaxT], f[MaxN], Size[MaxN], Ans[MaxQ]; bool isRoot[MaxT], Rev[MaxT]; struct ES
{
int u, v, w, Idx;
bool Del;
} E[MaxM]; struct QR
{
int f, x, y, Pos;
} Q[MaxQ]; inline bool CmpW(ES e1, ES e2)
{
return e1.w < e2.w;
} inline bool CmpIdx(ES e1, ES e2)
{
return e1.Idx < e2.Idx;
} inline bool CmpUV(ES e1, ES e2)
{
return (e1.u < e2.u) || (e1.u == e2.u && e1.v < e2.v);
} int FindIdx(int x, int y)
{
if (x > y) swap(x, y);
int l, r, mid;
l = 1; r = m;
while (l <= r)
{
mid = (l + r) >> 1;
if (E[mid].u == x && E[mid].v == y) break;
if ((E[mid].u < x) || (E[mid].u == x && E[mid].v < y)) l = mid + 1;
else r = mid - 1;
}
return mid;
} inline int Find(int x)
{
int i, j, k;
j = x;
while (j != f[j]) j = f[j];
i = x;
while (i != j)
{
k = i;
i = f[i];
f[k] = j;
}
return j;
} inline void UN(int x, int y)
{
if (Size[x] == Size[y]) ++Size[x];
if (Size[x] > Size[y]) f[y] = x;
else f[x] = y;
} /********************* LCT Begin *********************/ inline int gmax(int a, int b) {return V[a] > V[b] ? a : b;} inline void Update(int x)
{
T[x] = gmax(x, gmax(T[Son[x][0]], T[Son[x][1]]));
} inline void Reverse(int x)
{
Rev[x] = !Rev[x];
swap(Son[x][0], Son[x][1]);
} inline void PushDown(int x)
{
if (!Rev[x]) return;
Rev[x] = false;
if (Son[x][0]) Reverse(Son[x][0]);
if (Son[x][1]) Reverse(Son[x][1]);
} inline int GetDir(int x)
{
if (x == Son[Father[x]][0]) return 0;
else return 1;
} void Rotate(int x)
{
int y = Father[x], f;
PushDown(y); PushDown(x);
if (x == Son[y][0]) f = 1;
else f = 0;
if (isRoot[y])
{
isRoot[y] = false;
isRoot[x] = true;
}
else
{
if (y == Son[Father[y]][0]) Son[Father[y]][0] = x;
else Son[Father[y]][1] = x;
}
Father[x] = Father[y];
Son[y][f ^ 1] = Son[x][f];
if (Son[x][f]) Father[Son[x][f]] = y;
Son[x][f] = y;
Father[y] = x;
Update(y); Update(x);
} void Splay(int x)
{
int y;
while (!isRoot[x])
{
y = Father[x];
if (isRoot[y])
{
Rotate(x);
break;
}
if (GetDir(y) == GetDir(x)) Rotate(y);
else Rotate(x);
Rotate(x);
}
} int Access(int x)
{
int y = 0;
while (x != 0)
{
Splay(x);
PushDown(x);
if (Son[x][1]) isRoot[Son[x][1]] = true;
Son[x][1] = y;
if (y) isRoot[y] = false;
Update(x);
y = x;
x = Father[x];
}
return y;
} inline void Make_Root(int x)
{
int t = Access(x);
Reverse(t);
} void Link(int x, int y)
{
Make_Root(x);
Splay(x);
Father[x] = y;
} void Cut(int x, int y)
{
Make_Root(x);
Access(y);
Splay(y);
PushDown(y);
isRoot[Son[y][0]] = true;
Father[Son[y][0]] = 0;
Son[y][0] = 0;
Update(y);
} /********************* LCT End *********************/ int main()
{
scanf("%d%d%d", &n, &m, &q);
for (int i = 1; i <= m; ++i)
{
Read(E[i].u); Read(E[i].v); Read(E[i].w);
if (E[i].u > E[i].v) swap(E[i].u, E[i].v);
E[i].Del = false;
}
sort(E + 1, E + m + 1, CmpW); // by ES.w
for (int i = 1; i <= m; ++i)
{
E[i].Idx = i;
V[n + i] = E[i].w;
}
for (int i = 1; i <= n + m; ++i)
{
isRoot[i] = true;
Father[i] = 0;
T[i] = i;
}
sort(E + 1, E + m + 1, CmpUV); // by ES.u && ES.v
int t;
for (int i = 1; i <= q; ++i)
{
Read(Q[i].f); Read(Q[i].x); Read(Q[i].y);
if (Q[i].f == 2)
{
t = FindIdx(Q[i].x, Q[i].y);
E[t].Del = true;
Q[i].Pos = E[t].Idx;
}
else ++Top;
}
Tot = Top;
for (int i = 1; i <= n; ++i)
{
f[i] = i;
Size[i] = 1;
}
sort(E + 1, E + m + 1, CmpIdx); // by ES.Idx
int Cnt = 0, fx, fy;
for (int i = 1; i <= m; ++i)
{
if (E[i].Del) continue;
fx = Find(E[i].u); fy = Find(E[i].v);
if (fx == fy) continue;
UN(fx, fy);
Link(E[i].u, n + i); Link(E[i].v, n + i);
if (++Cnt == n - 1) break;
}
int CutE;
for (int i = q; i >= 1; --i)
{
Make_Root(Q[i].x);
t = Access(Q[i].y);
if (Q[i].f == 1) Ans[Top--] = V[T[t]];
else
{
if (E[Q[i].Pos].w >= V[T[t]]) continue;
CutE = T[t];
Cut(CutE, E[CutE - n].u); Cut(CutE, E[CutE - n].v);
Link(Q[i].x, n + Q[i].Pos); Link(Q[i].y, n + Q[i].Pos);
}
}
for (int i = 1; i <= Tot; ++i) printf("%d\n", Ans[i]);
return 0;
}
[BZOJ 2594] [Wc2006]水管局长数据加强版 【LCT】的更多相关文章
- BZOJ 2594: [Wc2006]水管局长数据加强版 [LCT kruskal]
2594: [Wc2006]水管局长数据加强版 Time Limit: 25 Sec Memory Limit: 128 MBSubmit: 2917 Solved: 918[Submit][St ...
- BZOJ 2594: [Wc2006]水管局长数据加强版( LCT )
离线然后就是维护加边的动态MST, Link cut tree秒掉..不过我写+调了好久...时间复杂度O(NlogN + MlogM) ------------------------------- ...
- 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]水管局长数据加强版(kruskal + LCT)
Description SC省MY市有着庞大的地下水管网络,嘟嘟是MY市的水管局长(就是管水管的啦),嘟嘟作为水管局长的工作就是:每天供水公司可能要将一定量的水从x处送往y处,嘟嘟需要为供水公司找到一 ...
- 【刷题】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)
论蒟蒻的自我修养T_T.. 和noi2014魔法森林基本一样...然而数据范围大得sxbk...UPD:这题如果用lct判联通的话可能会被卡到O(mlogm)..所以最好还是用并查集吧 一开始数组开太 ...
随机推荐
- Android.mk各种文件编译汇总
一.源代码编译 1.1 so预编译 LOCAL_PATH:= $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := libAppArea LOCAL ...
- [Usaco2006 Nov]Corn Fields牧场的安排 壮压DP
看到第一眼就发觉是壮压DP 然后就三进制枚举子集吧. 这题真是壮压入门好题... 对于dp[i][j] 表示第i行,j状态下前i行的分配方案数. 那么dp[i][j]肯定是从i-1行转过来的 那么由于 ...
- windows 下一个 easy_install 设备
下载安装python安装工具 1,方法是下载ez_setup.py后 2,在cmd下运行 python ez_setup.py.就可以自己主动安装setuptools 3,环境变量设置将 C:\Pro ...
- WTL的消息机制
Windows消息 众所周知,Windows消息有两种:队列话消息和非队列话消息.队列话消息是被Windows操作系统放入消息队列的,程序通过主消息循环不断的从消息队列中取出消息并分发到各自的窗体调用 ...
- CentOS7上GitLab的使用
生成SSH Keys 生成root账号的ssh key # ssh-keygen -t rsa -C "admin@example.com" 显示pub key的值 # cat ~ ...
- System Operations on AWS - Lab 6W - Using Auto Scaling (Windows)
创建你的一个web server,然后将这个实例制成你的AMI,通过启动配置生成一个Auto Scaling组(包括scale-in/scale-out策略),配置一台Load Balancer指向你 ...
- JQuery字符串替换replace方法
在日常的js开发中,常常会用到JQuery, 当要把字符串中的内容替换时,如果使用类似C#的string.replace方法,如下 var str='aabbccaa'; str=str.replac ...
- PHP之APC缓存详细介绍
1.APC缓存简介 APC,全称是Alternative PHP Cache,官方翻译叫"可选PHP缓存".它为我们提供了缓存和优化PHP的中间代码的框架. APC的缓存分两部分: ...
- Node.js + Express + Mongodb 开发搭建个人网站(三)
三.后台架构 1.在根目录下(和 views 文件夹同级)创建 lib 文件夹 以后所有后端内容 都是在这里写,分别创建三个文件夹 到 lib 目录下: mongo 放的是数据的存储 module ...
- YesNo列
比较,注意两边类型是否一致,以及boolean类型tostring之后的值 if(item["IsShow"].ToString() == "True")