BZOJ2333 [SCOI2011]棘手的操作 堆 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong
去博客园看该题解
题目传送门 - BZOJ2333
题意概括
有N个节点,标号从1到N,这N个节点一开始相互不连通。第i个节点的初始权值为a[i],接下来有如下一些操作:
U x y: 加一条边,连接第x个节点和第y个节点
A1 x v: 将第x个节点的权值增加v
A2 x v: 将第x个节点所在的连通块的所有节点的权值都增加v
A3 v: 将所有节点的权值都增加v
F1 x: 输出第x个节点当前的权值
F2 x: 输出第x个节点所在的连通块中,权值最大的节点的权值
F3: 输出所有节点中,权值最大的节点的权值
题解
我比较懒。
这是一道坑坑的细节题。
我调了大约6个小时。
对于A3,我们只需要保存一个全局变量,每次输出的时候加上去就可以了。
对于F3,我们只需要弄一个multiset就可以了。
对于U、A1、A2、F1、F2,就是左偏树的几个基本操作。
“嘴巴上AC就是简单”
代码
#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <set>
#include <vector>
using namespace std;
const int N=300005,Inf=1e8;
int n,m,addv=0;
struct Tree{
int fa,ls,rs,v,add,len;
Tree (){}
Tree (int a,int b,int c,int d,int e,int f){
fa=a,ls=b,rs=c,v=d,add=e,len=f;
}
}t[N];
multiset <int> rts;
void check0(){
t[0]=Tree(0,0,0,-Inf,0,-1);
}
int getroot(int x){
check0();
while (t[x].fa)
x=t[x].fa;
return x;
}
void del(int x){
rts.erase(rts.find(x));
}
void pushdown(int x){
check0();
int s,&add=t[x].add;
if (!t[x].add)
return;
s=t[x].ls;
if (s)
t[s].add+=add,t[s].v+=add;
s=t[x].rs;
if (s)
t[s].add+=add,t[s].v+=add;
add=0;
}
void pushadd(int x){
check0();
if (t[x].fa)
pushadd(t[x].fa);
pushdown(x);
}
int merge(int a,int b){
check0();
if (a==b)
return a;
if (a==0)
return b;
if (b==0)
return a;
if (t[a].v<t[b].v)
swap(a,b);
pushdown(a);
t[a].rs=merge(t[a].rs,b);
t[t[a].rs].fa=a;
if (t[t[a].ls].len<t[t[a].rs].len)
swap(t[a].ls,t[a].rs);
t[a].len=t[t[a].rs].len+1;
return a;
}
void change(int x,int val){
check0();
pushadd(x);
int rt=getroot(x),maxv=t[rt].v;
int fa=t[x].fa,s=merge(t[x].ls,t[x].rs);
if (fa==0&&s==0){
del(t[x].v);
t[x].v=val;
rts.insert(t[x].v);
return;
}
t[s].fa=0;
if (fa){
t[s].fa=fa;
if (t[fa].ls==x)
t[fa].ls=s;
else
t[fa].rs=s;
}
t[x]=Tree(0,0,0,val,0,0);
merge(x,getroot(s?s:fa));
int maxv_=t[getroot(x)].v;
if (maxv_!=maxv){
del(maxv);
rts.insert(maxv_);
}
}
int main(){
scanf("%d",&n);
check0();
rts.clear();
for (int i=1,v;i<=n;i++){
scanf("%d",&v);
t[i]=Tree(0,0,0,v,0,0);
rts.insert(v);
}
scanf("%d",&m);
while (m--){
char op[5],c1,c2;
int a,b;
scanf("%s",op),c1=op[0],c2=op[1];
if (c1=='U'){
scanf("%d%d",&a,&b);
a=getroot(a),b=getroot(b);
if (a==b)
continue;
if (t[a].v<t[b].v)
swap(a,b);
del(t[b].v);
merge(a,b);
}
else if (c1=='A'){
if (c2=='1')
scanf("%d%d",&a,&b),pushadd(a),change(a,t[a].v+b);
else if (c2=='2'){
scanf("%d%d",&a,&b);
a=getroot(a);
del(t[a].v);
t[a].add+=b,t[a].v+=b;
rts.insert(t[a].v);
}
else
scanf("%d",&a),addv+=a;
}
else {
if (c2=='1')
scanf("%d",&a),pushadd(a),printf("%d\n",t[a].v+addv);
else if (c2=='2')
scanf("%d",&a),printf("%d\n",t[getroot(a)].v+addv);
else
printf("%d\n",*--rts.end()+addv);
} }
return 0;
}
BZOJ2333 [SCOI2011]棘手的操作 堆 左偏树 可并堆的更多相关文章
- 【BZOJ2333】棘手的操作(左偏树,STL)
[BZOJ2333]棘手的操作(左偏树,STL) 题面 BZOJ上看把... 题解 正如这题的题号 我只能\(2333\) 神TM棘手的题目... 前面的单点/联通块操作 很显然是一个左偏树+标记 ( ...
- Luogu P3273 [SCOI2011]棘手的操作(左偏树)
什么恶心东西啊,又水又烦 两个可并堆维护即可 #include <cstdio> #include <iostream> #include <cstring> #i ...
- 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树|可并堆-左偏树)
2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...
- BZOJ2333 [SCOI2011]棘手的操作 【离线 + 线段树】
题目 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: 将第x个节点的权 ...
- [note]左偏树(可并堆)
左偏树(可并堆)https://www.luogu.org/problemnew/show/P3377 题目描述 一开始有N个小根堆,每个堆包含且仅包含一个数.接下来需要支持两种操作: 操作1: 1 ...
- bzoj2809 [Apio2012]dispatching——左偏树(可并堆)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2809 思路有点暴力和贪心,就是 dfs 枚举每个点作为管理者: 当然它的子树中派遣出去的忍者 ...
- [bzoj2333] [SCOI2011]棘手的操作 (可并堆)
//以后为了凑字数还是把题面搬上来吧2333 发布时间果然各种应景... Time Limit: 10 Sec Memory Limit: 128 MB Description 有N个节点,标号从1 ...
- bzoj千题计划218:bzoj2333: [SCOI2011]棘手的操作
http://www.lydsy.com/JudgeOnline/problem.php?id=2333 上次那个是线段树,再发一个左偏树 维护两种左偏树 第一种是对每个联通块维护一个左偏树 第二种是 ...
- 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树)
2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...
随机推荐
- 集大软件工程15级个人作业Week2
集大软件工程15级个人作业Week2 快速通读教材<构建之法>,并参照提问模板,提出5个问题. 在每个问题后面,请说明哪一章节的什么内容引起了你的提问,提供一些上下文 列出一些事例或资料, ...
- python时间序列画图plot总结
画图从直觉上来讲就是为了更加清晰的展示时序数据所呈现的规律(包括趋势,随时间变化的规律(一周.一个月.一年等等)和周期性规律),对于进一步选择时序分析模型至关重要.下面主要是基于pandas库总结一下 ...
- MySQL5.7使用错误解决:ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using password: NO)【取消或重设root密码】
解决方法: 最简单方法: ⑴打开mysql中的my.ini(如果没有就将my-default.ini复制一份,并修改为my.ini): ⑵在[mysqld]下面空白行直接添加skip-grant-ta ...
- 一个极好的JavaScript学习网址
JavaScript学习网址:https://javascript.info/ 网址界面如下,内容和排版都非常不错,内容也比较新,不像一些教程已经是好几年前的了.把这个教程浏览一遍,能够解答很多看代码 ...
- 用winhotkey添加属于自己的快捷键
需求 我要使用Win+N快捷键组合打开指定某个文件! 使用方法 打开winhotkey软件,做以下操作: 此刻,就可以用Win+N组合快捷键来打开指定目录了!
- Java9都快发布了,Java8的十大新特性你了解多少呢?
Java 9预计将于今年9月份发布,这是否会是一次里程碑式的版本,我们拭目以待.今天,我们先来复习一下2014年发布的Java 8的十大新特性. Java 8可谓是自Java 5以来最具革命性的版本了 ...
- iOS视频开发经验
iOS视频开发经验 手机比PC的优势除了便携外,我认为最重要的就是可以快速方便的创作多媒体作品.照片分享,语音输入,视频录制,地理位置.一个成功的手机APP从产品形态上都有这其中的一项或多项,比如in ...
- Java调整JVM内存大小——(八)
JVM内存大小是有限的,有的时候程序过大时候会导致超出最大内存,产生下面异常: Exception in thread "main" java.lang.OutOfMemoryEr ...
- 安装ClamAV对centos系统进行病毒查杀
安装ClamAV 1.安装epel源 yum install epel-release 在安装了EPEL源后,运行下面的命令安装ClamAV # yum install clamav-server c ...
- Mac安装Homebrew记录
在终端输入: ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install) ...