\[Preface
\]

之前用 treap 打,交了四遍才过。

自学了 fhq treap 后,才意识到是一道 fhq treap 板子题,直接码上,一遍就过。

本题解提供的是 fhq treap 做法(感谢 fhq 神犇),不太了解 fhq treap 的同学可以去了解一下 fhq treap ,挺好理解的。(至少要了解两种 split 和 merge 操作)

\[Description
\]

你需要维护一个可重集,支持 \(4\) 种操作:

I k 向集合内插入元素 \(k\) 。

A k 将集合内所有元素加上 \(k\) 。

S k 将集合内所有元素减去 \(k\) 。

F k 查询集合内第 \(k\) 大。

一共 \(n\) 次操作。

一开始给出一个下界 \(Minv\) ,表示集合内的所有元素必须大于等于 \(Minv\) ,在任何时刻,小于 \(Minv\) 的所有元素会被立刻删除。

最后还要输出被删除的元素个数。

\[Solution
\]

对于 I k 操作:

​ 首先先判断一下 \(k\) 是否大于等于 \(Minv\) ,之后就是经典的插入操作了。

​ 让原树按 \(k-1\) 分裂成两棵树 \(x,y\) ,新建值为 \(k\) 的节点 \(New\) ,合并 \(x,New,y\) 。

\(~\)

对于 A kS k 操作:

​ 考虑到 " AS 命令的总条数不超过 100 " ,想到了啥?暴力修改!

​ 显然集合内的所有元素加上 \(k\) 或减去 \(k\),fhq treap 树形态不变。

A k 操作就很好办了,集合内的所有元素加上 \(k\) 固然还是大于等于 \(Minv\) ,直接 \(dfs\) 暴力修改即可。

S k 操作之后还会导致一些元素小于 \(Minv\) ,这些元素在还没操作前是小于 \(Minv+k\) 的。

​ 那么我们可以将原树按 \(Minv+k-1\) 分裂成两棵树 \(x,y\) 。树 \(x\) 的所有元素在操作之后都是小于 \(Minv\) 的,那么此次被删除的元素个数即为 \(size[x]\) ,将其累加进答案,随后树 \(x\) 就没有用了,不用管它就行了。树 \(y\) 的所有元素在操作之后都是大于等于 \(Minv\) 的,直接 \(dfs\) 暴力修改,最后将树 \(y\) 当成原树就行了。

\(~\)

对于 F k 操作:

​ 经典的查询第 \(k\) 大。

​ 让原树按 \(size[root]-k\) 大小分裂成两棵树 \(x,y\) ,再让树 \(y\) 按 \(1\) 大小分裂成两棵树 \(y,z\) ,此时树 \(y\) 只有一个节点,这个节点就是我们要找的第 \(k\) 大了,最后记得合并回去。

​ 当然,在 " 按大小分裂 " 上做一点小改动,或是用一般 treap 的查询第 k 大也是可以的。

\(~\)

至此,此题得到完美解决。

\[Code
\]

#include<cstdio>
#include<cstdlib>
#include<algorithm> #define RI register int using namespace std; inline int read()
{
int x=0,f=1;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
} const int N=100100; int m,Minv; int tot,root;
struct treap{
int lc,rc;
int val,dat;
int size;
}t[N]; int New(int val)
{
tot++;
t[tot].lc=t[tot].rc=0;
t[tot].val=val;
t[tot].dat=rand();
t[tot].size=1;
return tot;
} void upd(int p)
{
t[p].size=t[t[p].lc].size+t[t[p].rc].size+1;
} void split_v(int p,int val,int &x,int &y)
{
if(!p)
x=y=0;
else
{
if(t[p].val<=val)
x=p,split_v(t[p].rc,val,t[p].rc,y);
else
y=p,split_v(t[p].lc,val,x,t[p].lc);
upd(p);
}
} void split_s(int p,int size,int &x,int &y)
{
if(!p)
x=y=0;
else
{
if(t[t[p].lc].size<size)
x=p,split_s(t[p].rc,size-t[t[p].lc].size-1,t[p].rc,y);
else
y=p,split_s(t[p].lc,size,x,t[p].lc);
upd(p);
}
} int merge(int p,int q)
{
if(!p||!q)
return p^q;
if(t[p].dat>t[q].dat)
{
t[p].rc=merge(t[p].rc,q),upd(p);
return p;
}
else
{
t[q].lc=merge(p,t[q].lc),upd(q);
return q;
}
} int x,y,z; void insert(int val)
{
split_v(root,val-1,x,y);
root=New(val);
root=merge(x,root);
root=merge(root,y);
} void dfs(int u,int val)
{
if(!u)return;
t[u].val+=val;
dfs(t[u].lc,val),dfs(t[u].rc,val);
} int GetValByRank(int rank)
{
if(t[root].size<rank)
return -1;
split_s(root,t[root].size-rank,x,y);
split_s(y,1,y,z);
int ans=t[y].val;
root=merge(x,y);
root=merge(root,z);
return ans;
} int ans; int main()
{
m=read(),Minv=read(); while(m--)
{
char op[2];scanf("%s",op);
int val=read(); switch(op[0])
{
case 'I':{
if(Minv<=val)
insert(val); break;
} case 'A':{
dfs(root,val); break;
} case 'S':{
split_v(root,Minv+val-1,x,root);
ans+=t[x].size; dfs(root,-val); break;
} case 'F':{
printf("%d\n",GetValByRank(val)); break;
}
}
} printf("%d\n",ans); return 0;
}

\[Thanks \ for \ watching
\]

题解 NOI2004【郁闷的出纳员】的更多相关文章

  1. [BZOJ1503][NOI2004]郁闷的出纳员

    [BZOJ1503][NOI2004]郁闷的出纳员 试题描述 OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是 ...

  2. BZOJ 1503: [NOI2004]郁闷的出纳员 splay

    1503: [NOI2004]郁闷的出纳员 Description OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作 ...

  3. [BZOJ1503][NOI2004]郁闷的出纳员 无旋Treap

    1503: [NOI2004]郁闷的出纳员 Time Limit: 5 Sec  Memory Limit: 64 MB Description OIER公司是一家大型专业化软件公司,有着数以万计的员 ...

  4. 【bzoj1503】[NOI2004]郁闷的出纳员

    1503: [NOI2004]郁闷的出纳员 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 13890  Solved: 5086[Submit][Stat ...

  5. bzoj 3224 NOI2004郁闷的出纳员

    NOI2004郁闷的出纳员 2013年12月26日6,1818 输入描述 Input Description 第一行有两个非负整数n和min.n表示下面有多少条命令,min表示工资下界. 接下来的n行 ...

  6. bzoj1503: [NOI2004]郁闷的出纳员(伸展树)

    1503: [NOI2004]郁闷的出纳员 题目:传送门 题解: 修改操作一共不超过100 直接暴力在伸展树上修改 代码: #include<cstdio> #include<cst ...

  7. BZOJ 1503: [NOI2004]郁闷的出纳员

    1503: [NOI2004]郁闷的出纳员 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 10526  Solved: 3685[Submit][Stat ...

  8. bzoj 1503: [NOI2004]郁闷的出纳员 Treap

    1503: [NOI2004]郁闷的出纳员 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 6263  Solved: 2190[Submit][Statu ...

  9. bzoj1503 [NOI2004]郁闷的出纳员(名次树+懒惰标记)

    1503: [NOI2004]郁闷的出纳员 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 8705  Solved: 3027[Submit][Statu ...

  10. BZOJ_1503_[NOI2004]郁闷的出纳员_权值线段树

    BZOJ_1503_[NOI2004]郁闷的出纳员_权值线段树 Description OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的 工资. ...

随机推荐

  1. 【转】15个超炫的HTML5效果

    英文原文:http://www.hongkiat.com/blog/15-html5-experiments/     翻译:iteye 乔布斯没有给Flash任何机会,微软新推出的Windows 8 ...

  2. kuangbin专题 专题九 连通图 POJ 3694 Network

    题目链接:https://vjudge.net/problem/POJ-3694 题目:给定一个连通图,求桥的个数,每次查询,加入一条边,问加入这条边后还有多少个桥. 思路:tarjan + 并查集 ...

  3. Spring--2.Spring之IOC--IOC容器的23个实验(2)

    Spring--2.Spring之IOC--IOC容器的23个实验(1) 中的所有实验我都是在同一个工程中进行的,从第十个实验开始,我将新建一个新的工程开始实验. 目前导包还是跟第一个项目一致,bea ...

  4. spring-boot内嵌三大容器https设置

    spring-boot内嵌三大容器https设置 spring-boot默认的内嵌容器为tomcat,除了tomcat之前还可以设置jetty和undertow. 1.设置https spring-b ...

  5. cogs 186. [USACO Oct08] 牧场旅行 树链剖分 LCA

    186. [USACO Oct08] 牧场旅行 ★★☆   输入文件:pwalk.in   输出文件:pwalk.out   逐字节对比时间限制:1 s   内存限制:128 MB n个被自然地编号为 ...

  6. 四、JVM之栈与栈帧

    栈: 1.又名堆栈,它是一种运算受限的线性表.其限制是仅允许在表的一端进行插入和删除运算.这一端被称为栈顶,相对地,把 另一端称为栈底.其特性是先进后出. 2.栈是线程私有的,生命周期跟线程相同,当创 ...

  7. 固定表头的table

    在前端的开发过程中,表格时经常使用的一种展现形式.在我的开发过程中,当数据过多时,最常用的一种方式就是分页,但是有些地方还是需要滚动.通常的table 会随着滚动,导致表头看不见.一下是我找到的一种固 ...

  8. 一些触发XSS的姿势(未完待续)

    本文对一些能触发XSS的方式进行记录与学习. HTML5特性向量 通过formaction属性进行XSS - 需要用户进行交互 formaction 属性规定当表单提交时处理输入控件的文件的 URL. ...

  9. java 自增/减运算符

    注意:python中没有 一.自增运算符 1.单独使用时,目的获取变量的值,前++和后++没有区别,使用后值都会递增一. 2.混合使用时,才有区别.前++,先加后用.后++,先用后加 二.自减运算符 ...

  10. Elastcisearch.Nest 7.x 系列`伪`官方翻译:通过 NEST 来快捷试用 Elasticsearch

    本系列已经已经全部完成,完整版可见:https://blog.zhuliang.ltd/categories/Elasticsearch/ 本系列博文是"伪"官方文档翻译(更加本土 ...