题目链接

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、并查集)的更多相关文章

  1. BZOJ 2959 长跑 (LCT+并查集)

    题面:BZOJ传送门 当成有向边做的发现过不去样例,改成无向边就忘了原来的思路.. 因为成环的点一定都能取到,我们把它们压成一个新点,权值为环上所有点的权值和 这样保证了图是一颗森林 每次询问转化为, ...

  2. BZOJ 2959: 长跑 LCT_并查集_点双

    真tm恶心...... Code: #include<bits/stdc++.h> #define maxn 1000000 using namespace std; void setIO ...

  3. 【bzoj2959】长跑 LCT+并查集

    题目描述 某校开展了同学们喜闻乐见的阳光长跑活动.为了能“为祖国健康工作五十年”,同学们纷纷离开寝室,离开教室,离开实验室,到操场参加3000米长跑运动.一时间操场上熙熙攘攘,摩肩接踵,盛况空前.为了 ...

  4. BZOJ 2959: 长跑 lct 双联通分量 并查集 splay

    http://www.lydsy.com/JudgeOnline/problem.php?id=2959 用两个并查集维护双联通分量的编号和合并. #include<iostream> # ...

  5. BZOJ 2959: 长跑 [lct 双连通分量 并查集]

    2959: 长跑 题意:字词加入边,修改点权,询问两点间走一条路径的最大点权和.不一定是树 不是树

  6. BZOJ2959长跑——LCT+并查集(LCT动态维护边双连通分量)

    题目描述 某校开展了同学们喜闻乐见的阳光长跑活动.为了能“为祖国健康工作五十年”,同学们纷纷离开寝室,离开教室,离开实验室,到操场参加3000米长跑运动.一时间操场上熙熙攘攘,摩肩接踵,盛况空前. 为 ...

  7. bzoj2959: 长跑 LCT+并查集+边双联通

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=2959 题解 调了半天,终于调完了. 显然题目要求是求出目前从 \(A\) 到 \(B\) 的可 ...

  8. bzoj2959: 长跑(LCT+并查集)

    题解 动态树Link-cut tree(LCT)总结 LCT常数大得真实 没有环,就是\(lct\)裸题吧 有环,我们就可以绕环转一圈,缩点 怎么搞? 当形成环时,把所有点的值全部加到一个点上,用并查 ...

  9. 【bzoj4998】星球联盟 LCT+并查集

    题目描述 在遥远的S星系中一共有N个星球,编号为1…N.其中的一些星球决定组成联盟,以方便相互间的交流.但是,组成联盟的首要条件就是交通条件.初始时,在这N个星球间有M条太空隧道.每条太空隧道连接两个 ...

随机推荐

  1. 【spring Boot】spring boot获取资源文件的三种方式【两种情况下】

    首先声明一点,springboot获取资源文件,需要看是 1>从spring boot默认的application.properties资源文件中获取 2>还是从自定义的资源文件中获取 带 ...

  2. luogu题解 P3709 【大爷的字符串题】

    题目链接: https://www.luogu.org/problemnew/show/P3709 思路: 首先我是没读懂题目的,浏览了讨论区的dalao发现才知道就是求区间众数的出现次数. 然后肯定 ...

  3. 12 Mysql之工作中常用操作

    Mysql 专题讲解 一.用户创建与权限管理 a)  创建和删除用户 创建用户: CREATE USER jack@localhost; UPDATE USER SET password=passwo ...

  4. WebStrom 中文显示异常中文变样乱码

    问题描述 WebStorm 编辑文件时中文显示异常,大小不一 菜单栏字体需要更换 解决方法 修改编辑器字体 菜单栏默认字体取消 设置效果 编辑文件时中英文显示 菜单栏 其他相关 关于编码格式,这里未做 ...

  5. vscode调试npm包技巧

    官网文档:https://code.visualstudio.com/docs/nodejs/nodejs-debugging node调试方法(日志和debuuger):https://blog.r ...

  6. vue学习(2)-过滤器

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. element-ui 中 el-table 根据scope.row行数据变化动态显示行内控件

    加入本行的数据为scope.row,其数据格式为 { "propertyId": 4, "propertyName": "标题", &quo ...

  8. JavaMaven【三、常用指令】

    mvn compile --编译,编译后生成target文件,里面包含classes mvn test --执行test,测试后在target下生成reports文件夹,测试报告 mvn packag ...

  9. 第四篇.python的基础

    目录 第四篇.python基础01 1. 变量 2. 常量 3. python变量内存管理 4. 变量的三个特征 5. 花式赋值 6. 注释 7. 数据类型基础 8. 数字类型 9. 字符串类型 10 ...

  10. Kubernetes介绍与核心组件

    Kubernetes是什么? Kubernetes是容器集群管理系统,是一个开源的平台,可以实现容器集群的自动化部署.自动扩缩容.维护等功能. Kubernetes 特点 可移植: 支持公有云,私有云 ...