[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开始去找目标长度,最少的就是正好 ...
- Android系统中setprop,getprop,watchprops命令的使用(转载)
转自:http://blog.csdn.net/yao_guet/article/details/6531241 在android系统中,有一些初始化的配置文件,例如: /init.rc /defau ...
- linux下系统调用劫持ioctl
实验环境:linux 2.6.32 64位系统 采用lkm(动态加载内核模块)方式劫持ioctl系统调用,系统调用过程如图所示(以open为例子) 实验代码:(头文件有不需要的,但是懒得改了,在系 ...
- 洛谷 P4012 深海机器人问题 【最大费用最大流】
和火星那个有点像,但是这个价值直接在路径上,不用拆点,对于每条价值为w的边(i,j),连接(i,j,1,w)(i,j,inf,0),表示价值只能取一次,然后连接源点和所有出发点(s,i,k,0),所有 ...
- 第四章vs2107 代码实际运用-后台权限管理讲解 创建角色
先看一下项目整体结构图: 实体类和数据操作都在前面用TT模板批量生产了.下面开始介绍权限代码这块的逻辑. 创建角色开始. 1. 角色的创建我们用到三张表 A.menuinfo(菜单表) role(角 ...
- Vue父子组件传值之——访问根组件$root、$parent、$children和$refs
Vue组件传值除了prop和$emit,我们还可以直接获取组件对象: 根组件: $root // 单一对象 表示当前组件树的根 Vue 实例,即new Vue({...根组件内容}).如果当前实例没有 ...
- [ZOJ1140]Courses 课程
Description 给出课程的总数P(1<=p<100),学生的总数N(1<=N<=300) 每个学生可能选了一门课程,也有可能多门,也有可能没有. 要求选出P个学生来组成 ...
- Hdu 5358 First One (尺取法+枚举)
题目链接: Hdu 5358 First One 题目描述: 数组a有n个元素,S[i,j]定义为a[i]+a[i+1]+.....+a[j],问:这个死东西等于多少? 解题思路: 二分肯定超,这个题 ...
- Codeforces Round #459 (Div. 2)C. The Monster
C. The Monster time limit per test 1 second memory limit per test 256 megabytes input standard input ...
- Appium教程---Client/Server Architecture
appium的核心其实是一个暴露了一系列REST API的server. 这个server的功能其实很简单:监听一个端口,然后接收由client发送来的command.翻译这些command,把这些c ...