BZOJ 2959 长跑 (LCT+并查集)
题面:BZOJ传送门
当成有向边做的发现过不去样例,改成无向边就忘了原来的思路..
因为成环的点一定都能取到,我们把它们压成一个新点,权值为环上所有点的权值和
这样保证了图是一颗森林
每次询问转化为,取出$a$到$b$这条链,求链上所有点的权值和
这实际是一个不删边的动态维护边双的过程
可以用$LCT$维护
加入一条边$<x,y>$时,我们取出链$x,y$
如果$x,y$原来不连通,把它们连上
否则说明$x,y$原来就联通的,连上这条边会成环,把$x,y$这条链上的点全都压成一个点,用并查集维护
每次$access$都在并查集里找父亲就行了
维护权值的时候细节比较多
#include <vector>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 150010
#define M1 (N1<<1)
#define il inline
#define idx(X) (X-'a')
using namespace std; char str[M1];
int n,m,de;
int a[N1],r[N1]; struct Union{
int fa[N1];
int findfa(int x)
{
int pre,y=x;
if(!x) return ;
while(fa[y]!=y) y=fa[y];
while(fa[x]!=y){ pre=fa[x]; fa[x]=y; x=pre; }
return y;
}
}U;
struct LCT{
int ch[N1][],fa[N1],rev[N1],sum[N1],stk[N1],tp;
il int idf(int x){ return (ch[fa[x]][]==x)?:;}
il void pushup(int x){ sum[x]=sum[ch[x][]]+sum[ch[x][]]+a[x]; }
il int isroot(int x){ return (ch[fa[x]][]!=x&&ch[fa[x]][]!=x)?:; }
il void revers(int x){ swap(ch[x][],ch[x][]),rev[x]^=; }
il void pushdown(int x)
{
if(rev[x])
{
if(ch[x][]) revers(ch[x][]);
if(ch[x][]) revers(ch[x][]);
rev[x]^=;
}
}
il void rot(int x)
{
int y=fa[x],ff=fa[y],px=idf(x),py=idf(y);
if(!isroot(y)) ch[ff][py]=x; fa[x]=ff;
fa[ch[x][px^]]=y; ch[y][px]=ch[x][px^];
fa[y]=x; ch[x][px^]=y;
pushup(y),pushup(x);
}
void splay(int x)
{
int y=x; stk[++tp]=x;
while(!isroot(y)){stk[++tp]=fa[y],y=fa[y];}
while(tp){pushdown(stk[tp--]);}
while(!isroot(x))
{
y=fa[x];
if(isroot(y)) rot(x);
else if(idf(y)==idf(x)) rot(y),rot(x);
else rot(x),rot(x);
}
}
void access(int x)
{
for(int y=;x;y=x,fa[x]=U.findfa(fa[x]),x=fa[x])
splay(x),ch[x][]=y,pushup(x);
}
void mkroot(int x)
{
access(x);
splay(x);
revers(x);
}
int findroot(int x)
{
access(x); splay(x);
while(ch[x][]) pushdown(ch[x][]),x=ch[x][];
splay(x); return x;
}
void split(int x,int y)
{
mkroot(x);
access(y);
splay(y);
}
void dfs(int x,int root)
{
U.fa[x]=root;
if(ch[x][]) dfs(ch[x][],root);
if(ch[x][]) dfs(ch[x][],root);
}
int link(int x,int y)
{
/*if(x==4&&y==1)
de=1; */
mkroot(x); access(y); splay(y);
int f=findroot(y);
if(f==x){ //merge
dfs(f,f); a[f]=sum[f];
ch[f][]=ch[f][]=;
}else{ //link
fa[x]=y;
}
}
int query(int x,int y)
{
split(x,y);
if(findroot(y)!=x) return -;
else return sum[x];
}
}lct; int main()
{
scanf("%d%d",&n,&m);
int i,j,A,B,x,y,v,fl,ans=,id=,de;
for(i=;i<=n;i++)
scanf("%d",&a[i]), lct.sum[i]=a[i], r[i]=a[i], U.fa[i]=i;
while(m--)
{
scanf("%d%d%d",&fl,&A,&B);
if(fl==)
{
x=U.findfa(A); y=U.findfa(B);
lct.link(x,y);
}
if(fl==)
{
x=U.findfa(A); lct.splay(x);
lct.sum[x]+=B-r[A];
a[x]+=B-r[A];
r[A]=B;
}
if(fl==)
{
x=U.findfa(A); y=U.findfa(B);
printf("%d\n",lct.query(x,y));
}
}
return ;
}
BZOJ 2959 长跑 (LCT+并查集)的更多相关文章
- BZOJ 2959: 长跑 LCT_并查集_点双
真tm恶心...... Code: #include<bits/stdc++.h> #define maxn 1000000 using namespace std; void setIO ...
- 【bzoj2959】长跑 LCT+并查集
题目描述 某校开展了同学们喜闻乐见的阳光长跑活动.为了能“为祖国健康工作五十年”,同学们纷纷离开寝室,离开教室,离开实验室,到操场参加3000米长跑运动.一时间操场上熙熙攘攘,摩肩接踵,盛况空前.为了 ...
- BZOJ 2959 长跑 (LCT、并查集)
题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=2959 题解 真是被这题搞得心态大崩--调了7个小时--然而并查集都能写成\(O(n^2) ...
- 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条太空隧道.每条太空隧道连接两个 ...
随机推荐
- 哈哈,找到一种方式来简单模拟EXTJS中与服务器的AJAX交互啦。
一直在测试客户端的EXTJS,但遇到服务器端就麻烦了,要建库,要写JSON,要有HTTP返回值. 今天测试了一个简单的方法,经过测试是OK了. 那,就是Python的SimpleHTTPServer模 ...
- CSDN日报20170416 ——《为什么程序猿话少钱多死得早?》
[程序人生]为什么程序猿话少钱多死得早? 作者:文奇 我在想,程序猿都是话少吗?不一定吧.像我和我的同学.都是话非常多啊. 可是经历过非常多事的如今.再想想,发现事实的确如此.程序猿确实话少. 我是一 ...
- [LeetCode]Wildcard Matching 通配符匹配(贪心)
一開始採用递归写.TLE. class Solution { public: bool flag; int n,m; void dfs(int id0,const char *s,int id1,co ...
- CefSharp 设置cookie
设置cookie var cookieManager = CefSharp.Cef.GetGlobalCookieManager(); await cookieManager.SetCookieAsy ...
- Linux 服务具体解释
acpid ACPI(全 称 Advanced Configuration and Power Interface)服务是电源管理接口. 建议全部的笔记本用户开启它. 一些server可能不须要 ac ...
- C语言数组和函数实例练习(一)
C语言的数组和函数部分的知识,在语法上和Java语法是有所相似的,这里只通过实例总结一些自己感觉需要理解的部分知识. 1.数组 数组中的元素具有相同的数据类型:数组一旦创建,不能被改变:数组中元素在内 ...
- hdoj--1533--Going Home(KM)
Going Home Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Tota ...
- Truck History --hdoj
Truck History Time Limit : 4000/2000ms (Java/Other) Memory Limit : 131072/65536K (Java/Other) Tota ...
- N数码问题的启发式搜索算法--A*算法python实现
一.启发式搜索:A算法 1)评价函数的一般形式 : f(n) = g(n) + h(n) g(n):从S0到Sn的实际代价(搜索的横向因子) h(n):从N到目标节点的估计代价,称为启发函数(搜索的纵 ...
- 在ubuntu中安装Markdown神器Typora
title: 在ubuntu中安装Markdown神器Typora toc: false date: 2018-09-01 17:48:15 categories: methods tags: ubu ...