【bzoj2333】 SCOI2011—棘手的操作
http://www.lydsy.com/JudgeOnline/problem.php?id=2333 (题目链接)
题意
N个节点维护一些操作。。
Solution
我们用可并大根堆进行维护。
对于每个连通块建一个局部可并堆,因为要询问全局最大值,所以还要对全局建一个全局可并堆记录之前局部可并堆堆顶元素。
U:合并x所在的堆以及y所在的堆,并在全局堆中删除合并前的局部堆堆顶元素,因为它合并以后已经不是其连通块的堆顶了。
A1:在堆中删除,更新后再加入堆
A2:找到其堆顶,对堆顶进行修改并打上标记
A3:对全局都打个标记即可
F1:将标记下传后输出
F2:找到其所在的堆顶输出
F3:输出全局堆的堆顶
UPD:这个做法被×了,暴力找堆顶复杂度不对,得写启发式合并或者线段树,右转LCF。。当然拿来练练手也是可以的,反正能过数据
细节
码农题就是细节多
代码
// bzoj2333
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#define LL long long
#define inf 1<<30
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; const int maxn=300010;
struct heap {int val,tag,l,r,fa;}q[maxn],qq[maxn];
int n,m,rt,Tag;
char op[10]; void pushdown(heap *q,int k) { //标记下传
q[q[k].l].val+=q[k].tag;q[q[k].l].tag+=q[k].tag;
q[q[k].r].val+=q[k].tag;q[q[k].r].tag+=q[k].tag;
q[k].tag=0;
}
int merge(heap *q,int x,int y) { //合并
if (x==0 || y==0) return x+y;
if (q[x].val<q[y].val) swap(x,y);
if (q[x].tag) pushdown(q,x);
q[x].r=merge(q,q[x].r,y);
q[q[x].r].fa=x;
swap(q[x].l,q[x].r);
return x;
}
int find(heap *q,int x) { //寻找堆顶并下传标记,注意下传标记和向上查询的顺序
int tmp=x;
if (q[x].fa) tmp=find(q,q[x].fa);
if (q[x].tag) pushdown(q,x);
return tmp;
}
int del(heap *q,int x) { //删除
int f=find(q,x);
int tmp=merge(q,q[x].l,q[x].r);
if (q[q[x].fa].l==x) q[q[x].fa].l=tmp;
else q[q[x].fa].r=tmp;
q[tmp].fa=q[x].fa;
return f==x ? (tmp ? find(q,tmp) : 0) : f; //返回删除后该堆的堆顶,此处不是很好处理,最好画个图理解一下
}
int build() { //对全局建堆
queue<int> Q;
for (int i=1;i<=n;i++) Q.push(i),qq[i]=q[i];
while (Q.size()>1) {
int x=Q.front();Q.pop();
int y=Q.front();Q.pop();
Q.push(merge(qq,x,y));
}
return Q.front();
}
void newq(heap *q,int x,int val) { //新建元素
q[x].l=q[x].r=q[x].fa=0;
q[x].val=val;
}
int main() {
scanf("%d",&n);q[0].val=-inf;
for (int i=1;i<=n;i++) scanf("%d",&q[i].val);
rt=build();
scanf("%d",&m);
for (int x,y,i=1;i<=m;i++) {
scanf("%s",op);
if (op[0]=='U') {
scanf("%d%d",&x,&y);
int r1=find(q,x),r2=find(q,y);
if (r1!=r2) {
if (merge(q,r1,r2)==r1) rt=del(qq,r2);
else rt=del(qq,r1);
}
}
if (op[0]=='A') {
if (op[1]=='1') {
scanf("%d%d",&x,&y);
rt=del(qq,find(q,x));
int k=del(q,x);
newq(q,x,q[x].val+y);
k=merge(q,k,x);
newq(qq,k,q[k].val);
rt=merge(qq,k,rt);
}
if (op[1]=='2') {
scanf("%d%d",&x,&y);
int f=find(q,x);
q[f].val+=y;q[f].tag+=y;
rt=del(qq,f);
newq(qq,f,qq[f].val+y);
rt=merge(qq,rt,f);
}
if (op[1]=='3') scanf("%d",&y),Tag+=y;
}
if (op[0]=='F') {
if (op[1]=='1') {
scanf("%d",&x);
find(q,x),printf("%d\n",q[x].val+Tag);
}
if (op[1]=='2') {
scanf("%d",&x);
printf("%d\n",q[find(q,x)].val+Tag);
}
if (op[1]=='3') printf("%d\n",qq[rt].val+Tag);
}
}
return 0;
}
【bzoj2333】 SCOI2011—棘手的操作的更多相关文章
- 真--可并堆模板--BZOJ2333: [SCOI2011]棘手的操作
n<=300000个点,开始是独立的,m<=300000个操作: 方法一:单点修改.查询,区间修改.查询?等等等等这里修改是块修改不是连续的啊,那就让他连续呗!具体方法:离线后,每次连接两 ...
- BZOJ2333 [SCOI2011]棘手的操作 堆 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2333 题意概括 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i ...
- [bzoj2333] [SCOI2011]棘手的操作 (可并堆)
//以后为了凑字数还是把题面搬上来吧2333 发布时间果然各种应景... Time Limit: 10 Sec Memory Limit: 128 MB Description 有N个节点,标号从1 ...
- bzoj千题计划217:bzoj2333: [SCOI2011]棘手的操作
http://www.lydsy.com/JudgeOnline/problem.php?id=2333 读入所有数据,先模拟一遍所有的合并操作 我们不关心联通块长什么样,只关心联通块内有谁 所以可以 ...
- 2019.01.17 bzoj2333: [SCOI2011]棘手的操作(启发式合并)
传送门 启发式合并菜题. 题意:支持与连通块有关的几种操作. 要求支持连边,单点修改,连通块修改,全局修改和单点查值,连通块查最大值和全局最大值. 我们对每个连通块和答案用可删堆维护最大值,然后用启发 ...
- BZOJ2333:[SCOI2011]棘手的操作(Splay)
Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: ...
- BZOJ2333 [SCOI2011]棘手的操作 【离线 + 线段树】
题目 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: 将第x个节点的权 ...
- bzoj2333 [SCOI2011]棘手的操作(洛谷3273)
题目描述 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作:U x y: 加一条边,连接第x个节点和第y个节点A1 x v: 将第x个节点的权 ...
- bzoj2333 [SCOI2011]棘手的操作
用set维护每个联通块里的最值,multiset维护所有块里的最值,并查集维护连通性,然后随便搞搞就行了,合并时候采用启发式合并.复杂度O(nlognlogn),大概勉强过的程度,反正跑的很慢就是了. ...
- (右偏树)Bzoj2333: [SCOI2011]棘手的操作
题面 戳我 Sol 右偏树滑稽+并查集 再在全局开一个可删除的堆(priority_queue) 注意细节 # include <bits/stdc++.h> # define RG re ...
随机推荐
- 分享最近写的 两条sql语句
1. 搭建基本环境 插入测试数据 insert into jgdm (jgdm,jgmc) values('12300000000','河南省');insert into jgdm (jgdm,jg ...
- XCODE6 提交至 App Store
新到一个公司,以前的苹果开发人员离职,临时接手他的苹果代码,需要修改并上线到APP STORE. xcode6.0升级到最新的6.1后, 发现各种坑 1. 路径配置不对, 这个是个人习惯问题,之前的 ...
- 解决js浮点数计算bug
1.加 function add(a, b) { var c, d, e; try { c = a.toString().split(".")[1].length; } catch ...
- 【BZOJ1483】【链表启发式合并】梦幻布丁
Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2,2,1的四个布丁一共有3段颜色. Input 第 ...
- linux常用命令(查看某些软件是否已安装)
查看imap是否已安装 rpm -qa | grep imap 以下为未安装的情形: 检查是否已安装sendmail: rpm -qa | grep sendmail 以下为已安装的返回:
- linuxC编程实战 my_server.c例子问题总结
今天看linux C 编程实战的my_server例子时,敲到这段代码,对其父子进程关闭socket 进行close调用产生疑问 如图中标注的三个close socket,思考子进程通信结束 关闭自己 ...
- jquery的select元素和option的相关操作
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- java 按天创建文件夹
按天创建文件夹,也就是每天创建一个,适合上传文件服务使用,文件数量较多时可以按文件夹区分. public static final String FMT = "yyyy-MM-dd" ...
- js+css3 动画数字累加
css: .kk{ width:100px; height:100px; display:inline-block; color:red; text-align:center; position: r ...
- [BZOJ 1692] [Usaco2007 Dec] 队列变换 【后缀数组 + 贪心】
---恢复内容开始--- 题目链接:BZOJ - 1692 题目分析 首先,有个比较简单的贪心思路:如果当前剩余字符串的两端字母不同,就选取小的字母,这样显然是正确的. 然而若两端字母相同,我们怎么选 ...