【BZOJ2333】【SCOI2011】棘手的操作 treap合并
题目大意
有\(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\):输出所有节点中,权值最大的节点的权值。
\(n,m\leq 300000,-1000\leq v,a_i\leq 1000\)
题解
如果没有点修改就可以用可合并堆维护。但是现在有点修改,这样就需要遍历\(x\)到根。然而可合并堆的深度是\(O(n)\)的,这样做会直接TLE。
那么还有没有什么其他的数据结构能支持合并和修改权值呢?
答案是:有。可以用线段树或非旋treap。合并均摊是\(O(\log n)\)的。我选择了treap。
treap合并在我的博客的另外一篇里面有写。
再拿一棵树维护所有treap的最大值的最大值。
用一个数维护全部节点的权值同时加了多少。输出时把这个加到答案上面去就可以了。
时间复杂度:\(O((n+m)\log n)\)
UPD:直接用treap的merge就行了。
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
#include<cmath>
#include<functional>
#include<set>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
void sort(int &a,int &b)
{
if(a>b)
swap(a,b);
}
void open(const char *s)
{
#ifndef ONLINE_JUDGE
char str[100];
sprintf(str,"%s.in",s);
freopen(str,"r",stdin);
sprintf(str,"%s.out",s);
freopen(str,"w",stdout);
#endif
}
struct node
{
int f,ls,rs;
int k,v,t,s;
int sz;
int u;
};
node a[300010];
int cnt=0;
int newnode(int u,int v)
{
int x=++cnt;
a[x].u=u;
a[x].v=a[x].s=v;
a[x].sz=1;
return x;
}
void add(int x,int v)
{
a[x].v+=v;
a[x].s+=v;
a[x].t+=v;
}
void push(int x)
{
if(a[x].t)
{
if(a[x].ls)
add(a[x].ls,a[x].t);
if(a[x].rs)
add(a[x].rs,a[x].t);
a[x].t=0;
}
}
void mt(int x)
{
a[x].s=a[x].v;
if(a[x].ls)
a[x].s=max(a[x].s,a[a[x].ls].s);
if(a[x].rs)
a[x].s=max(a[x].s,a[a[x].rs].s);
a[x].sz=a[a[x].ls].sz+a[a[x].rs].sz+1;
}
void pushdown(int x)
{
if(a[x].f)
pushdown(a[x].f);
push(x);
}
void pushup(int x)
{
mt(x);
if(a[x].f)
pushup(a[x].f);
}
int merge(int x,int y)
{
if(!x||!y)
return x+y;
if(a[x].k<a[y].k)
{
push(x);
a[x].rs=merge(a[x].rs,y);
mt(x);
if(a[x].rs)
a[a[x].rs].f=x;
return x;
}
else
{
push(y);
a[y].ls=merge(x,a[y].ls);
mt(y);
if(a[y].ls)
a[a[y].ls].f=y;
return y;
}
}
pii split(int x,int v)
{
if(!x)
return pii();
push(x);
if(v<a[x].u)
{
pii s=split(a[x].ls,v);
a[x].ls=s.second;
if(s.second)
a[s.second].f=x;
s.second=x;
a[x].f=0;
return s;
}
else
{
pii s=split(a[x].rs,v);
a[x].rs=s.first;
if(s.first)
a[s.first].f=x;
s.first=x;
a[x].f=0;
return s;
}
}
int join(int x,int y)
{
if(!x||!y)
return x+y;
if(a[x].k>a[y].k)
swap(x,y);
push(x);
pii s=split(y,a[x].v);
a[x].ls=merge(a[x].ls,s.first);
a[x].rs=merge(a[x].rs,s.second);
if(a[x].ls)
a[a[x].ls].f=x;
if(a[x].rs)
a[a[x].rs].f=x;
mt(x);
return x;
}
int f[300010];
int find(int x)
{
return f[x]==x?x:f[x]=find(f[x]);
}
int rt[300010];
multiset<int> s;
int main()
{
open("bzoj2333");
int n;
scanf("%d",&n);
int i,x;
for(i=1;i<=n;i++)
{
scanf("%d",&x);
rt[i]=newnode(i,x);
f[i]=i;
s.insert(x);
}
char op[5];
int y,v;
int all=0;
int m;
int ans;
scanf("%d",&m);
for(i=1;i<=m;i++)
{
scanf("%s",op);
if(op[0]=='U')
{
scanf("%d%d",&x,&y);
x=find(x);
y=find(y);
if(x==y)
continue;
f[x]=y;
s.erase(s.find(min(a[rt[x]].s,a[rt[y]].s)));
rt[y]=join(rt[x],rt[y]);
}
else if(op[0]=='A'&&op[1]=='1')
{
scanf("%d%d",&x,&v);
y=find(x);
s.erase(s.find(a[rt[y]].s));
pushdown(x);
a[x].v+=v;
pushup(x);
s.insert(a[rt[y]].s);
}
else if(op[0]=='A'&&op[1]=='2')
{
scanf("%d%d",&x,&v);
x=find(x);
s.erase(s.find(a[rt[x]].s));
add(rt[x],v);
s.insert(a[rt[x]].s);
}
else if(op[0]=='A'&&op[1]=='3')
{
scanf("%d",&y);
all+=y;
}
else if(op[1]=='1')
{
scanf("%d",&x);
pushdown(x);
ans=a[x].v+all;
printf("%d\n",ans);
}
else if(op[1]=='2')
{
scanf("%d",&x);
x=find(x);
ans=a[rt[x]].s+all;
printf("%d\n",ans);
}
else
{
ans=*s.rbegin()+all;
printf("%d\n",ans);
}
}
return 0;
}
【BZOJ2333】【SCOI2011】棘手的操作 treap合并的更多相关文章
- 真--可并堆模板--BZOJ2333: [SCOI2011]棘手的操作
n<=300000个点,开始是独立的,m<=300000个操作: 方法一:单点修改.查询,区间修改.查询?等等等等这里修改是块修改不是连续的啊,那就让他连续呗!具体方法:离线后,每次连接两 ...
- bzoj2333[SCOI2011]棘手的操作 洛谷P3273 [SCOI2011]棘手的操作
2333? 先记一下吧,这题现在全部都是照着题解做的,因为怎么改都改不出来,只好对着题解改,以后还要再做过 以后再也不用指针了!太恶心了!空指针可不止直接特判那么简单啊,竟然还要因为空指针写奇怪的分类 ...
- 2019.01.17 bzoj2333: [SCOI2011]棘手的操作(启发式合并)
传送门 启发式合并菜题. 题意:支持与连通块有关的几种操作. 要求支持连边,单点修改,连通块修改,全局修改和单点查值,连通块查最大值和全局最大值. 我们对每个连通块和答案用可删堆维护最大值,然后用启发 ...
- [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 读入所有数据,先模拟一遍所有的合并操作 我们不关心联通块长什么样,只关心联通块内有谁 所以可以 ...
- BZOJ2333 [SCOI2011]棘手的操作 堆 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2333 题意概括 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i ...
- BZOJ2333:[SCOI2011]棘手的操作(Splay)
Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: ...
- 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),大概勉强过的程度,反正跑的很慢就是了. ...
随机推荐
- Leetcode -- 394. Decode String
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- Vue Router 路由实现原理
一.概念 通过改变 URL,在不重新请求页面的情况下,更新页面视图. 二.实现方式 更新视图但不重新请求页面,是前端路由原理的核心之一,目前在浏览器环境中这一功能的实现主要有2种方式: 1.Hash ...
- 295B - Greg and Graph (floyd逆序处理)
题意:给出任意两点之间的距离,然后逐个删除这些点和与点相连的边,问,在每次删除前的所有点对的最短距离之和 分析:首先想到的是floyd,但是如果从前往后处理,复杂度是(500)^4,超时,我们从后往前 ...
- 软件工程(FZU2015) 赛季得分榜,第10回合(alpha冲刺)
SE_FZU目录:1 2 3 4 5 6 7 8 9 10 11 12 13 积分规则 积分制: 作业为10分制,练习为3分制:alpha30分: 团队项目分=团队得分+个人贡献分 个人贡献分: 个人 ...
- C#复习笔记(5)--C#5:简化的异步编程(异步编程的深入分析)
首先,阐明一下标题的这个“深入分析”起得很惭愧,但是又不知道该起什么名字,这个系列也主要是做一些复习的笔记,供自己以后查阅,如果能够帮助到别人,那自然是再好不过了. 然后,我想说的是异步方法的状态机真 ...
- [转帖]AMOLED的技术和OLED有哪些联系和区别
AMOLED的技术和OLED有哪些联系和区别 https://display.ofweek.com/2018-06/ART-11000-2300-30243226.html 硬件资料 导读: ?虽然L ...
- 使用cmd命令删除文件夹下所有文件
rmdir 删除整个目录 好比说我要删除 222 这个目录下的所有目录和档案,这语法就是: rmdir /s/q 222 其中: /s 是代表删除所有子目录跟其中的档案. /q 是不要它在删除档案或目 ...
- MySQL Connector/ODBC 2.50 (MyODBC 2.50)连接方式
一.MySQL Connector/ODBC 2.50 (MyODBC 2.50)连接方式 1.本地数据库连接Driver={MySQL};Server=localhost;Option=16834; ...
- ES6 & Map & hashMap
ES6 & Map & hashMap 01 two-sum https://leetcode.com/submissions/detail/141732589/ hashMap ht ...
- python数据结构与算法第十四天【二分查找】
1.二分查找的原理 对于已经排序的列表进行最快速度的查找 2. 代码实现 (1)递归实现 def binary_search(alist, item): if len(alist) == 0: ret ...