BZOJ 2959 长跑 (LCT、并查集)
题目链接
https://www.lydsy.com/JudgeOnline/problem.php?id=2959
题解
真是被这题搞得心态大崩……调了7个小时……然而并查集都能写成\(O(n^2)\)的我还能怪谁呢
显然要把每个边双连通分量缩成点,点权为边双连通分量内所有点点权和,然后答案就等于两点路径上点权和
现在需要用LCT维护,就比较麻烦
大概是一边LCT一边使用并查集分别维护连通块和边双连通分量
加边时,若两点不联通,则link
, 然后在维护连通块的并查集里并起来
若两点联通但不在同一边双中,则把这两个点路径上的所有边双缩到一起(其实就是“删除点”),顺便加入到边双连通分量的并查集中
这个可以通过把路径的splay提取出来进行DFS实现,因为每个点只会被删一次所以复杂度正确
但是这里由于缩点,我们每次在lct中访问父亲的时候要求它树上父亲在并查集里的代表元素。。。所以很容易写错
时间复杂度\(O(n\log n\alpha(n))\).
代码
#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<cassert>
#include<ctime>
#define llong long long
using namespace std;
const int N = 1.5e5;
struct SplayNode
{
int son[2],fa,sum,val,rev;
} spl[N+3];
int uf1[N+3],uf2[N+3];
int stk[N+3];
int a[N+3];
int n,q;
inline int read()
{
int ret = 0; char ch = getchar();
while(ch < '0' || ch > '9') ch = getchar();
while(ch >= '0' && ch <= '9') ret = (ret << 3) + (ret << 1) + ch - '0' , ch = getchar();
return ret;
}
int findfa(int id,int u)
{
if(id==0)
{
int i = u;
while(u!=uf1[u]) {u = uf1[u];}
while(uf1[i]!=u)
{
int j = uf1[i]; uf1[i] = u; i = j;
}
}
else
{
int i = u;
while(u!=uf2[u]) {u = uf2[u];}
while(uf2[i]!=u)
{
int j = uf2[i]; uf2[i] = u; i = j;
}
}
return u;
}
bool isroot(int u) {int uu = findfa(1,spl[u].fa); return spl[uu].son[0]!=u && spl[uu].son[1]!=u;}
bool sondir(int u) {return u==spl[findfa(1,spl[u].fa)].son[1];}
void pushup(int u)
{
spl[u].sum = spl[spl[u].son[0]].sum+spl[u].val+spl[spl[u].son[1]].sum;
}
void pushdown(int u)
{
int ls = spl[u].son[0],rs = spl[u].son[1];
if(spl[u].rev)
{
spl[u].rev = 0;
if(ls)
{
swap(spl[ls].son[0],spl[ls].son[1]);
spl[ls].rev ^= 1;
}
if(rs)
{
swap(spl[rs].son[0],spl[rs].son[1]);
spl[rs].rev ^= 1;
}
}
}
void rotate(int u)
{
int x = findfa(1,spl[u].fa),y = findfa(1,spl[x].fa); bool dir = sondir(u)^1;
if(!isroot(x)) {spl[y].son[sondir(x)] = u;}
spl[u].fa = y;
spl[x].son[dir^1] = spl[u].son[dir];
if(spl[x].son[dir^1]) {spl[spl[x].son[dir^1]].fa = x;}
spl[u].son[dir] = x; spl[x].fa = u;
pushup(x);
}
void splaynode(int u)
{
int x = u,tp = 0,y;
while(!isroot(x)) {tp++; stk[tp] = x; x = findfa(1,spl[x].fa);}
pushdown(x);
while(tp) {pushdown(stk[tp]); tp--;}
while(!isroot(u))
{
x = findfa(1,spl[u].fa),y = findfa(1,spl[x].fa);
if(!isroot(x)) {sondir(x)^sondir(u) ? rotate(u) : rotate(x);}
rotate(u);
}
pushup(u);
}
void access(int u)
{
for(int i=0; u; i=u,u=findfa(1,spl[u].fa))
{
splaynode(u);
spl[u].son[1] = i; pushup(u);
}
}
void makeroot(int u)
{
access(u); splaynode(u);
spl[u].rev ^= 1; swap(spl[u].son[0],spl[u].son[1]);
}
void link(int u,int v)
{
makeroot(u); spl[u].fa = v;
}
void dfs(int u,int u0)
{
uf2[u] = u0;
pushdown(u);
if(spl[u].son[0]) dfs(spl[u].son[0],u0);
if(spl[u].son[1]) dfs(spl[u].son[1],u0);
}
int main()
{
scanf("%d%d",&n,&q);
for(int i=1; i<=n; i++) uf1[i] = uf2[i] = i;
for(int i=1; i<=n; i++) a[i] = read(),spl[i].val = spl[i].sum = a[i];
while(q--)
{
int opt; opt = read();
if(opt==1)
{
int u,v; u = read(),v = read();
int uu = findfa(0,u),vv = findfa(0,v);
if(uu!=vv)
{
link(findfa(1,u),findfa(1,v));
uf1[uu] = vv;
}
else
{
uu = findfa(1,u),vv = findfa(1,v);
makeroot(uu); access(vv); splaynode(vv);
spl[vv].val = spl[vv].sum;
dfs(vv,vv);
spl[vv].son[0] = 0;
}
}
else if(opt==2)
{
int u,x; u = read(),x = read(); int delta = x-a[u]; a[u] = x;
int uu = findfa(1,u); splaynode(uu);
spl[uu].val += delta; spl[uu].sum += delta;
}
else if(opt==3)
{
int u,v; u = read(),v = read();
int uu = findfa(1,u),vv = findfa(1,v);
int uuu = findfa(0,uu),vvv = findfa(0,vv);
if(uuu!=vvv) {puts("-1"); continue;}
else
{
makeroot(uu); access(vv); splaynode(vv);
printf("%d\n",spl[vv].sum);
}
}
}
return 0;
}
BZOJ 2959 长跑 (LCT、并查集)的更多相关文章
- BZOJ 2959 长跑 (LCT+并查集)
题面:BZOJ传送门 当成有向边做的发现过不去样例,改成无向边就忘了原来的思路.. 因为成环的点一定都能取到,我们把它们压成一个新点,权值为环上所有点的权值和 这样保证了图是一颗森林 每次询问转化为, ...
- BZOJ 2959: 长跑 LCT_并查集_点双
真tm恶心...... Code: #include<bits/stdc++.h> #define maxn 1000000 using namespace std; void setIO ...
- 【bzoj2959】长跑 LCT+并查集
题目描述 某校开展了同学们喜闻乐见的阳光长跑活动.为了能“为祖国健康工作五十年”,同学们纷纷离开寝室,离开教室,离开实验室,到操场参加3000米长跑运动.一时间操场上熙熙攘攘,摩肩接踵,盛况空前.为了 ...
- BZOJ 2959: 长跑 lct 双联通分量 并查集 splay
http://www.lydsy.com/JudgeOnline/problem.php?id=2959 用两个并查集维护双联通分量的编号和合并. #include<iostream> # ...
- BZOJ 2959: 长跑 [lct 双连通分量 并查集]
2959: 长跑 题意:字词加入边,修改点权,询问两点间走一条路径的最大点权和.不一定是树 不是树
- BZOJ2959长跑——LCT+并查集(LCT动态维护边双连通分量)
题目描述 某校开展了同学们喜闻乐见的阳光长跑活动.为了能“为祖国健康工作五十年”,同学们纷纷离开寝室,离开教室,离开实验室,到操场参加3000米长跑运动.一时间操场上熙熙攘攘,摩肩接踵,盛况空前. 为 ...
- bzoj2959: 长跑 LCT+并查集+边双联通
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=2959 题解 调了半天,终于调完了. 显然题目要求是求出目前从 \(A\) 到 \(B\) 的可 ...
- bzoj2959: 长跑(LCT+并查集)
题解 动态树Link-cut tree(LCT)总结 LCT常数大得真实 没有环,就是\(lct\)裸题吧 有环,我们就可以绕环转一圈,缩点 怎么搞? 当形成环时,把所有点的值全部加到一个点上,用并查 ...
- 【bzoj4998】星球联盟 LCT+并查集
题目描述 在遥远的S星系中一共有N个星球,编号为1…N.其中的一些星球决定组成联盟,以方便相互间的交流.但是,组成联盟的首要条件就是交通条件.初始时,在这N个星球间有M条太空隧道.每条太空隧道连接两个 ...
随机推荐
- 垃圾分类API
###自从上月开始实行垃圾分类,整个上海人都已经“疯掉”了.那么,我们应该怎么识别什么是“垃圾”呢? 这里,我提供了并整理了几个api,其中包括图像识别,垃圾分类查找,根据关键字或者图片的内容来进行查 ...
- 怎样安装 cnpm 并切换到淘宝镜像?
如果不使用 vpn , 在国内直接使用 npm 的官方镜像会很慢,这里推荐使用淘宝 NPM 镜像.淘宝 NPM 镜像是一个完整的 npmjs.org 镜像,可以用此代替官方版本(只读). 操作方法如下 ...
- 解决tomcat端口冲突
1.根据8080端口号查找占用8080端口的进程 netstat -ano|findstr " 进程id:6352 2.根据进程ID查找进程名字 tasklist|findstr " ...
- sql使用临时表循环
code CREATE PROCEDURE sp_Update_Blogger_Blog_ArticleCount AS BEGIN declare @account varchar(); --博主账 ...
- golang(10):web开发 & 连接数据库
http编程 ) Go原生支持 http : import ("net/http") ) Go 的 http 服务性能和 nginx 比较接近 ) 几行代码就可以实现一个 web ...
- win10上使用php与python实现与arduino串口通信
注意: php 需要php7,安装及开启php_dio.dll com口按照实际的进行设置,如果不知道可以打开arduino编辑器进行查看 可以与用户实现命令行交互,但是效率过慢,不清楚如何优化,使用 ...
- 嵌套For循环性能优化
请对以下的代码进行优化 for (int i = 0; i < 1000; i++) for (int j = 0; j < 100; j++) for (int k = 0; k < ...
- Redis之各版本特性
1.Redis2.6 Redis2.6在2012年正是发布,经历了17个版本,到2.6.17版本,相对于Redis2.4,主要特性如下: 1)服务端支持Lua脚本. 2)去掉虚拟内存相关功能. 3)放 ...
- 使用javascript和jquery获取类方法
1.本质区别 jquery是一个javascript库.jquery是一个基于javascript语言的框架,本质上就是javascript. 2.代码编写的差异 jquery大大简化了JavaScr ...
- RAD Studio 10.3.2七大新功能介绍
RAD Studio 10.3.2七大新功能 Delphi支持macOS 64位应用的开发. C++Builder中Windows 64位平台支持C ++ 17特性. C ++ LSP 代码洞察改进. ...