[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: 加一条边 ...
随机推荐
- CodeForces 721B Passwords (水题)
题意:给定 n 个密码,你要按长度不递减的顺序进行尝试,问你最多和最少试多少次可能找出密码,每尝试 k 次错误的,就要等5秒. 析:我们只要把长度全都统计下来,然后从1开始去找目标长度,最少的就是正好 ...
- _bzoj1026 [SCOI2009]windy数【数位dp】
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1026 数位dp果断记忆化搜索,代码量少~ 程序里我用11代表前导零. #include &l ...
- [USACO 2012 Mar Gold] Large Banner
传送门:http://www.usaco.org/index.php?page=viewproblem2&cpid=127 又是一道这种题目,遇到一次跪一次,这次终于硬着头皮看懂了题解,但是谢 ...
- 牛客小白月赛5-D-阶乘(求n内每个数的阶乘相乘后尾数为0的个数)
题目描述 输入描述: 输入数据共一行,一个正整数n,意义如“问题描述”. 输出描述: 输出一行描述答案:一个正整数k,表示S的末尾有k个0 输入例子: 10 输出例子: 7 --> 示例1 输入 ...
- 题解报告:hdu 1235 统计同成绩学生人数
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1235 Problem Description 读入N名学生的成绩,将获得某一给定分数的学生人数输出. ...
- Matlab实现图像分割 分类: 图像处理 2014-06-14 21:31 662人阅读 评论(1) 收藏
下面使用极小值点阈值选取方法,编写MATLAB程序实现图像分割的功能. 极小值点阈值选取法即从原图像的直方图的包络线中选取出极小值点, 并以极小值点为阈值将图像转为二值图像 clear all; cl ...
- java实现九九乘法表
public class Demo { public static void main(String[] args) { for (int i = 1; i < 10; i++) { ...
- AJPFX总结在循环中break与continue的区别
相信刚学编程的人很容易被break,continue这两个关键词搞混淆了,两者都有跳出循环的意思,但是他们到底有什么区别呢?其实很简单,break是结束整个循环体,continue是结束当前这一单次循 ...
- javax.xml.bind.UnmarshalException: 意外的元素 (uri:"", local:"xml")。所需元素为(none)
将xml转换为object时候报错:javax.xml.bind.UnmarshalException: 意外的元素 (uri:"", local:"xml") ...
- addslashes,stripslashes
官方介绍: (PHP 4, PHP 5) addslashes — 使用反斜线引用字符串 返回字符串,该字符串为了数据库查询语句等的需要在某些字符前加上了反斜线.这些字符是单引号(’).双引号(”). ...