题目描述

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. 学完自动化测试,用小技能做了点兼职刷弹幕,小赚10W

    大家好,今天又给大家带来了Python爬虫的分享,继续来研究一下虎牙平台的爬虫. 起因 我冒出有一个很有趣的想法,就是,我们可以使用selenium来完成虎牙自动化登录,并且自动给主播发送弹幕功能的程 ...

  2. 第三节:Centos下安装Mysql5.6数据库

    1.下载mysql5.6版本 [官网-需要什么版本自己去找]https://dev.mysql.com/downloads/mysql/5.6.html [版本]linux通用 cd /usr/loc ...

  3. kafka笔记——kafka启动

    1. 准备 阿里云Linux服务器一个(Centos7) 腾讯云Linux服务器一个(CentOs7) zookeeper,kafka压缩包 Java环境配置好 要死....脚本之家 2. 安装 zo ...

  4. 机器学习 | 简介推荐场景中的协同过滤算法,以及SVD的使用

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是机器学习专题的第29篇文章,我们来聊聊SVD在上古时期的推荐场景当中的应用. 推荐的背后逻辑 有没有思考过一个问题,当我们在淘宝或者是 ...

  5. mac下高效安装 homebrew 及完美避坑姿势 (亲测有效)

    世上无难事,只要找到 Homebrew 的正确安装方式. Homebrew 是什么 Homebrew是 mac的包管理器,仅需执行相应的命令,就能下载安装需要的软件包,可以省掉自己去下载.解压.拖拽( ...

  6. 什么是 PHP 过滤器?

    PHP 过滤器 PHP 过滤器用于验证和过滤来自非安全来源的数据,比如用户的输入. 什么是 PHP 过滤器? PHP 过滤器用于验证和过滤来自非安全来源的数据. 测试.验证和过滤用户输入或自定义数据是 ...

  7. PHP strncasecmp() 函数

    实例 比较两个字符串(不区分大小写): <?php高佣联盟 www.cgewang.comecho strncasecmp("Hello world!","hell ...

  8. ajax模拟表单提交,后台使用npoi实现导入操作 方式一

    页面代码: <form id="form1" enctype="multipart/form-data"> <div style=" ...

  9. luogu P2354 [NOI2014]随机数生成器 贪心 卡空间 暴力

    LINK:随机数生成器 观察数据范围还是可以把矩阵给生成出来的. 考虑如何求出答案.题目要求把选出的数字从小到大排序后字典序尽可能的小 实际上这个类似于Mex的问题. 所以要从大到小选数字 考虑选择一 ...

  10. CF R638 div2 F Phoenix and Memory 贪心 线段树 构造 Hall定理

    LINK:Phoenix and Memory 这场比赛标题好评 都是以凤凰这个单词开头的 有凤来仪吧. 其实和Hall定理关系不大. 不过这个定理有的时候会由于 先简述一下. 对于一张二分图 左边集 ...