题目描述

OIER公司是一家大型专业化软件公司,有着数以万计的员工。作为一名出纳员,我的任务之一便是统计每位员工的工资。这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复无常,经常调整员工的工资。如果他心情好,就可能把每位员工的工资加上一个相同的量。反之,如果心情不好,就可能把他们的工资扣除一个相同的量。我真不知道除了调工资他还做什么其它事情。

工资的频繁调整很让员工反感,尤其是集体扣除工资的时候,一旦某位员工发现自己的工资已经低于了合同规定的工资下界,他就会立刻气愤地离开公司,并且再也不会回来了。每位员工的工资下界都是统一规定的。每当一个人离开公司,我就要从电脑中把他的工资档案删去,同样,每当公司招聘了一位新员工,我就得为他新建一个工资档案。

老板经常到我这边来询问工资情况,他并不问具体某位员工的工资情况,而是问现在工资第k多的员工拿多少工资。每当这时,我就不得不对数万个员工进行一次漫长的排序,然后告诉他答案。

好了,现在你已经对我的工作了解不少了。正如你猜的那样,我想请你编一个工资统计程序。怎么样,不是很困难吧?

如果某个员工的初始工资低于最低工资标准,那么将不计入最后的答案内

思路

用 splay 维护即可,介绍 splay 的文章可参考 『这里』

#include <cstdio>
const int maxn = 100000 + 10;
int n,m,num,ans,delta,root,father[maxn],ch[maxn][2],key[maxn],cnt[maxn],size[maxn];
inline void clear(int x) { ch[x][0] = ch[x][1] = key[x] = cnt[x] = size[x] = father[x] = 0; }
inline bool getfa(int x) { return ch[father[x]][1] == x; }
inline void Update(int x) {
if (x) {
size[x] = cnt[x];
if (ch[x][0]) size[x] += size[ch[x][0]];
if (ch[x][1]) size[x] += size[ch[x][1]];
}
}
inline void rotate(int x) {
int old = father[x],oldfa = father[father[x]],w = getfa(x);
ch[old][w] = ch[x][w^1]; father[ch[old][w]] = old;
ch[x][w^1] = old; father[old] = x;
father[x] = oldfa;
if (oldfa) ch[oldfa][ch[oldfa][1] == old] = x;
Update(old);
Update(x);
}
inline void splay(int x) {
for (int fa;fa = father[x];rotate(x))
if (father[fa]) rotate(getfa(fa) == getfa(x) ? fa : x);
root = x;
}
inline void Insert(int x) {
if (!root) {
num++;
ch[num][0] = ch[num][1] = father[num] = 0;
key[num] = x;
cnt[num] = size[num] = 1;
root = num;
return;
}
int now = root,fa = 0;
while (true) {
if (key[now] == x) {
cnt[now]++;
Update(now);
Update(fa);
splay(now);
break;
}
fa = now;
now = ch[now][key[now] < x];
if (!now) {
num++;
ch[num][0] = ch[num][1] = 0;
father[num] = fa;
cnt[num] = size[num] = 1;
ch[fa][key[fa] < x] = num;
key[num] = x;
Update(fa);
splay(num);
break;
}
}
}
inline int getnum(int x) {
int now = root,ans = 0;
while (true) {
if (key[now] > x) now = ch[now][0];
else {
if (ch[now][0]) ans += size[ch[now][0]];
if (key[now] == x) {
splay(now);
return ans+1;
}
ans += cnt[now];
now = ch[now][1];
}
}
}
inline int find(int x) {
int now = root;
while (true) {
if (ch[now][0] && x <= size[ch[now][0]]) now = ch[now][0];
else {
int tmp = cnt[now];
if (ch[now][0]) tmp += size[ch[now][0]];
if (x <= tmp) return key[now];
x -= tmp;
now = ch[now][1];
}
}
}
inline void Delete(int x) {
getnum(x);
if (cnt[root] > 1) {
cnt[root]--;
Update(root);
return;
}
if (!ch[root][0] && !ch[root][1]) {
clear(x);
root = 0;
return;
} else if (!ch[root][0] || !ch[root][1]) {
int oldroot = root;
root = ch[root][!ch[root][0]];
father[root] = 0;
clear(oldroot);
return;
}
int l = ch[root][0],oldroot = root;
while (ch[l][1]) l = ch[l][1];
splay(l);
ch[root][1] = ch[oldroot][1];
father[ch[oldroot][1]] = root;
clear(oldroot);
Update(root);
} int main() {
scanf("%d%d\n",&n,&m);
while(n--) {
char op;
int k;
scanf("%s%d",&op,&k);
if (op == 'I' && k >= m) Insert(k-delta);
if (op == 'F') printf("%d\n",k <= size[root] ? find(size[root]-k+1)+delta : -1);
if (op == 'A') delta += k;
if (op == 'S') {
int tmp = size[root];
delta -= k;
for (int i = 1;i <= tmp;i++)
if (find(1)+delta < m) {
Delete(find(1));
ans++;
}
}
}
printf("%d",ans);
return 0;
}

【NOI2004】郁闷的出纳员 - Splay的更多相关文章

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

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

  2. NOI2004 郁闷的出纳员 Splay

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

  3. 【BZOJ1503】 [NOI2004]郁闷的出纳员 splay

    splay模板题,都快把我做忧郁了. 由于自己调两个坑点. 1.删除时及时updata 2.Kth 考虑k满足该点的条件即r->ch[1]->size+1<=k && ...

  4. 洛谷P1486 [NOI2004]郁闷的出纳员(splay)

    题目描述 OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复无常,经常调整员工的工资 ...

  5. BZOJ1503 [NOI2004]郁闷的出纳员 splay

    原文链接http://www.cnblogs.com/zhouzhendong/p/8086240.html 题目传送门 - BZOJ1503 题意概括 如果某一个员工的工资低于了min,那么,他会立 ...

  6. 洛谷.1486.[NOI2004]郁闷的出纳员(Splay)

    题目链接 /* BZOJ1503: 3164kb 792ms/824ms(新建节点) 洛谷 : 3.06mb 320ms/308ms(前一个要慢wtf 其实都差不多,但前者好写) 四种操作: A:所有 ...

  7. bzoj1503[NOI2004]郁闷的出纳员——Splay

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1503 好奇怪呀!为什么而TLE? 各种修改终于卡时过了.可是大家比我快多了呀?难道是因为自己 ...

  8. BZOJ[NOI2004]郁闷的出纳员 | Splay板子题

    题目: 洛谷也能评测....还有我wa了10多次的记录233 题解: 不要想得太复杂,搞一个全局变量记录一下工资的改变量Delta,这样可以等询问的时候就输出val+Delta,然后插入的时候插入x- ...

  9. BZOJ_1503 [NOI2004]郁闷的出纳员 【Splay树】

    一 题面 [NOI2004]郁闷的出纳员 二 分析 模板题. 对于全部员工的涨工资和跌工资,可以设一个变量存储起来,然后在进行删除时,利用伸展树能把结点旋转到根的特性,能够很方便的删除那些不符合值的点 ...

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

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

随机推荐

  1. 最小割&网络流应用

    重要链接 基础部分链接 : 二分图 & 网络流初步 zzz大佬博客链接 : 网络流学习笔记 重点内容:最小割二元关系新解(lyd's ppt) 题目:网络流相关题目 lyd神犇课件链接 : 网 ...

  2. 超简单的jq图片上传

    <label class="file_img" for="file_imgs"> <input class="file_imgs&q ...

  3. IntelliJ IDEA:文件的路径本该是”\“,却变成了”¥“

    修改字体导致的, 有些字体中是用¥替换掉\的,所以,修改到合适的字体就好了

  4. python为什么这么火?里面肯定是有原因的

    因为人生苦短要用python啊! 看完本文,你将在结尾得到本文的一个福利彩蛋 你瞧瞧其他语言之父... Java之父——James Gosling PHP之父 ——Rasmus Lerdorf Obj ...

  5. w10查看wifi密码

    1.选择网络和Internet设置 右键单击网络连接图标,选择“打开网络和Internet设置”. 2.选择网络和共享中心

  6. layui常用插件(二) 时间插件

    日期和时间 html <div class="layui-inline"> <!-- 注意:这一层元素并不是必须的 --> <input type=& ...

  7. laravel 上线部署最佳实践

    nginx  配置 listen 80 default_server; server_name xxxx; index index.php index.html;    优先 index.php ro ...

  8. spring 循环依赖的一次 理解

    前言: 在看spring 循环依赖的问题中,知道原理,网上一堆的资料有讲原理. 但今天在看代码过程中,又产生了疑问. 疑问点如下: // 疑问点: 先进行 dependon 判断String[] de ...

  9. luogu P5826 【模板】子序列自动机 主席树 vector 二分

    LINK:子序列自动机 想了一些很有趣的做法. dp 容易看出 f[i][j]表示前i个数匹配了j个数的dp 不过复杂度很高. 贪心 容易想到匹配的时候每个数字尽量往前匹配 这样显然是最优的 复杂度Q ...

  10. docker 容器使用 systemctl 命令是报错

    看了许多解决方案,但是对于新手来说并不友好,不是特别清楚 报错内容: System has not been booted with systemd as init system (PID 1). C ...