[SCOI2011]棘手的操作(可并堆/并查集/线段树)
过于棘手
但这题真的很水的说
毕竟写啥都能过
常见思路:
①:由于不强制在线,所以重新编号之后线段树维护
②:用各种可以高速合并的数据结构,比如可并堆,可并平衡树啥的
讲一种无脑算法:
对于$F1$,并查集乱搞
对于$F2$,用可并堆维护连通块里的值,并维护对应的时间,如果堆顶访问到旧的值直接抛出
对于$F3$,用全局堆维护每个连通块的最大值的集合并维护对应的时间,如果堆顶访问到旧的值直接抛出
$A1$:单点修改时在所在可并堆里插入一个新的,维护修改时间,同时在全局堆里插入一个新的该可并堆的最大值
$A2$:在堆上打tag,在全局堆中插入新的最大值
$A3$:维护一个全局变量
$U$:略
好气啊将近二百行可并堆写的一点问题没有结果十多行并查集出了三处锅...
#include<cstdio> #include<queue> #include<algorithm> using std::max; using std::priority_queue; using std::swap; ; int n,v[N],del[N],Del; ]; int lta[N],ltb[N]; struct shino { int id,v,t; shino(){} shino(int a,int b,int c){id=a,v=b,t=c;} bool friend operator < (shino x,shino y){return x.v<y.v;} }g; struct merheap { ],tag[N<<],son[N<<][],fa[N<<],kr,yop[N<<],rt[N<<]; shino el[N<<]; int find(int x) { if(fa[x]==x) return x; else return fa[x]=find(fa[x]); } void fuckdown(int x) { if(tag[x]) { ]; if(k){el[k].v+=tag[x];tag[k]+=tag[x];} k=son[x][]; if(k){el[k].v+=tag[x];tag[k]+=tag[x];} tag[x]=; } } int makenew(shino sn) { kr++; el[kr]=sn; fa[kr]=kr; return kr; } int merge(int x,int y) { if(!x) return y; if(!y) return x; fuckdown(x); fuckdown(y); if(el[x]<el[y]) swap(x,y); son[x][]=merge(son[x][],y); fa[son[x][]]=x; ]]<dis[son[x][]]) swap(son[x][],son[x][]); dis[x]=dis[son[x][]]+; return x; } void push(int x,shino ei) { int tmp=makenew(ei); int xx=find(x); rt[x]=merge(xx,tmp); } void pop(int x) { int xx=find(x); fuckdown(xx); ],son[xx][]); fa[xx]=tmp; ]) fa[son[xx][]]=tmp; ]) fa[son[xx][]]=tmp; rt[x]=tmp; } shino top(int x) { x=find(x); return el[x]; } void update(int x) { int xx=find(x); while(xx) { xx=find(xx); g=top(xx); if(g.t!=lta[g.id]) pop(xx); else break; } } void add(int x,int vi) { int xx=find(x); el[xx].v+=vi; tag[xx]+=vi; } void init() { ;i<=n;i++) { makenew(shino(i,v[i],)); rt[i]=i; } } }rk; int fa[N]; bool isrt[N]; void domerge(int x,int y); priority_queue<shino> q; void find(int x) { if(fa[x]!=fa[fa[x]]) find(fa[x]),v[x]+=del[fa[x]]; fa[x]=fa[fa[x]]; } void merge(int x,int y,int tt) { find(x),find(y); int fx=fa[x],fy=fa[y]; if(fx==fy) return; domerge(fx,fy); q.push(shino(fy,rk.top(fy).v,tt)); ltb[fy]=tt; fa[fx]=fy; del[fx]-=del[fy]; v[fx]+=del[fx]; isrt[fx]=; } void domerge(int x,int y) { int xx=rk.find(x),yy=rk.find(y); int tmp=rk.merge(xx,yy); rk.fa[xx]=rk.fa[yy]=rk.rt[y]=tmp; } void fuckdate() { while(!q.empty()) { g=q.top(); if(!isrt[g.id]||g.t!=ltb[g.id]) q.pop(); else break; } } int main() { scanf("%d",&n); ;i<=n;i++) scanf(; int T; scanf("%d",&T); rk.init(); ;i<=n;i++) q.push(shino(i,v[i],)); int xin,yin,vin; ;t<=T;t++) { scanf("%s",op); ]=='U') { scanf("%d%d",&xin,&yin); merge(xin,yin,t); }]=='A') { ]==') { scanf("%d%d",&xin,&vin); find(xin); v[xin]+=vin; rk.push(fa[xin],shino(xin,v[xin]+del[fa[xin]],t)); lta[xin]=t; rk.update(fa[xin]); q.push(shino(fa[xin],rk.top(fa[xin]).v,t)); ltb[fa[xin]]=t; }]==') { scanf("%d%d",&xin,&vin); find(xin); del[fa[xin]]+=vin; rk.add(fa[xin],vin); q.push(shino(fa[xin],rk.top(fa[xin]).v,t)); ltb[fa[xin]]=t; }else { scanf("%d",&vin); Del+=vin; } }]=='F') { ]==') { scanf("%d",&xin); find(xin); printf("%d\n",v[xin]+del[fa[xin]]+Del); }]==') { scanf("%d",&xin); find(xin); rk.update(fa[xin]); printf("%d\n",rk.top(fa[xin]).v+Del); }else { fuckdate(); g=q.top(); printf("%d\n",g.v+Del); } } } ; }
巨佬您txdy
题外话:你这题解也太水了吧
rkk:因为确实水死了啊...都快成板子了...
[SCOI2011]棘手的操作(可并堆/并查集/线段树)的更多相关文章
- 【bzoj2333】 [SCOI2011]棘手的操作 可并堆+lazy标记
2016-05-31 21:45:41 题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2333 (学习了黄学长的代码 有如下操作: U x y ...
- BZOJ 2333: [SCOI2011]棘手的操作 可并堆 左偏树 set
https://www.lydsy.com/JudgeOnline/problem.php?id=2333 需要两个结构分别维护每个连通块的最大值和所有连通块最大值中的最大值,可以用两个可并堆实现,也 ...
- 【bzoj2333】[SCOI2011]棘手的操作 可并堆+STL-set
UPD:复杂度是fake的...大家还是去写启发式合并吧. 题目描述 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条 ...
- [bzoj2333] [SCOI2011]棘手的操作 (可并堆)
//以后为了凑字数还是把题面搬上来吧2333 发布时间果然各种应景... Time Limit: 10 Sec Memory Limit: 128 MB Description 有N个节点,标号从1 ...
- BZOJ 2333 [SCOI2011]棘手的操作 (可并堆)
码农题.. 很显然除了两个全局操作都能用可并堆完成 全局最大值用个multiset记录,每次合并时搞一搞就行了 注意使用multiset删除元素时 如果直接delete一个值,会把和这个值相同的所有元 ...
- bzoj 2333 [SCOI2011]棘手的操作 —— 可并堆
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2333 稍微复杂,参考了博客:http://hzwer.com/5780.html 用 set ...
- 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树|可并堆-左偏树)
2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...
- 真--可并堆模板--BZOJ2333: [SCOI2011]棘手的操作
n<=300000个点,开始是独立的,m<=300000个操作: 方法一:单点修改.查询,区间修改.查询?等等等等这里修改是块修改不是连续的啊,那就让他连续呗!具体方法:离线后,每次连接两 ...
- 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树)
2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...
随机推荐
- Linux - 常用命令索引
文件和目录操作 pwd - 显示当前所在的位置 cd - 切换目录 文件过滤及内容编辑 ... 文本处理三剑客 ... 系统信息查询与搜索文件 ... 文件备份与压缩 ... 用户管理与信息查询 .. ...
- 洛谷 P2593 [ZJOI2006]超级麻将【dp】
设f[i][j][k][0/1]表示选到i时,i-1选j张,i选k张,之前选的所有牌是否选择了对子 然后分情况讨论转移即可 #include<iostream> #include<c ...
- bzoj 2132 圈地计划【最小割+dinic】
对于网格图,尤其是这种要求相邻各自不同的,考虑黑白染色 对于这张染色后图来说: 对于每个黑格: 表示初始时选择商业区: s点向它连商业区收益的流量,它向t点连工业区收益的流量: 割断S侧的边说明反悔, ...
- bzoj 4540: [Hnoi2016]序列【单调栈+线段树】
强烈安利:http://blog.csdn.net/qq_34637390/article/details/51313126 这篇讲标记讲的非常好,这个标记非常神奇-- 首先last表示扫描到last ...
- 【插件开发】—— 7 SWT布局详解,不能再详细了!
前文回顾: 1 插件学习篇 2 简单的建立插件工程以及模型文件分析 3 利用扩展点,开发透视图 4 SWT编程须知 5 SWT简单控件的使用与布局搭配 6 SWT复杂空间与布局搭配 前面几篇都提到了S ...
- 关于Http的小常识(转载,仅供参考)
HTTP请求头提供了关于请求,响应或者其他的发送实体的信息.HTTP的头信息包括通用头.请求头.响应头和实体头四个部分.每个头域由一个域名,冒号(:)和域值三部分组成. 通用头标:即可用于请求,也可用 ...
- Sgu294He's Circles
Description 有一个长度为N的环,上面写着'X'和'E',问本质不同的环有多少种.(N不超过200000). Input The input file contains a single i ...
- BZOJ 3224 SBT 普通平衡树
复习了一下SBT的模板,但是BZOJ不知道为什么注册不了,所以就没交,测了样例能过! #include <bits/stdc++.h> #include<algorithm> ...
- 暴力 ZOJ 1403 Safecracker
题目传送门 /* 暴力:纯暴力,在家水水 */ #include <cstdio> #include <cstring> #include <algorithm> ...
- linux下的c程序排版工具:indent 分类: linux 2014-06-14 20:05 720人阅读 评论(0) 收藏
Linux下有一个方便的c语言程序排版工具,只要选择恰当的参数,可以轻易地使自己的程序具有统一的风格. 当然首先要安装indent,执行命令:apt-get install indent indent ...