题目描述

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. Python 正则表达式简单了解

    match 从字符串的开始匹配  如果开头不符合要求  就会报错 search  用字符串里的每一个元素  去匹配找的元素 1.匹配单个字符 \d 数字 \D 非数字 . 匹配任意字符 除了\n [] ...

  2. 使用jwt进行token认证

    简单说明:最近在搞权限这一块的东西,需要用到jwt进行token认证,才有了如下的demo演示   具体细节可以看gitbug,噗,不是bug是hub  github地址:https://github ...

  3. Mybatis(一)Mybatis简介与入门程序

    Mybatis简介: MyBatis是一个优秀的持久层框架,它对jdbc的操作数据库的过程进行封装,使开发者只需要关注 SQL 本身,而不需要花费精力去处理例如注册驱动.创建connection.创建 ...

  4. windows系统远程修改密码

    1.需求:公司需要短时间.批量修改一些windows系统的管理员密码: 2.准备工作: a.下载软件:链接:https://pan.baidu.com/s/1kV52DqE1_4siPuxS5Mosc ...

  5. Spring集成CXF发布WebService并在客户端调用

    Spring集成CXF发布WebService 1.导入jar包 因为官方下载的包里面有其他版本的sprring包,全导入会产生版本冲突,所以去掉spring的部分,然后在项目根目录下新建了一个CXF ...

  6. phpcms视频模块实现列表页打开内容页直接播放视频

    摘自phpcms论坛 原链接地址:http://bbs.phpcms.cn/thread-557691-1-1.html 之前下载研究过“化蝶自在飞”开发的视频模型,发现功能不错,但唯一的缺憾是,我想 ...

  7. Day02_WebCrawler(网络爬虫)

    学于黑马和传智播客联合做的教学项目 感谢 黑马官网 传智播客官网 微信搜索"艺术行者",关注并回复关键词"webcrawler"获取视频和教程资料! b站在线视 ...

  8. PHP getimagesizefromstring - 获取图片信息函数

    getimagesizefromstring — 从字符串中获取图像尺寸信息.高佣联盟 www.cgewang.com 语法 array getimagesizefromstring ( string ...

  9. 4.28 省选模拟赛模拟赛 最佳农场 二维卷积 NTT

    第一次遇到二维卷积 不太清楚是怎么做的. 40分暴力比对即可. 对于行为或者列为1时 容易想到NTT做快速匹配.然后找答案即可. 考虑这是一个二维的比对过程. 设\(f_{i,j}\)表示以i,j为右 ...

  10. 三类安装VMTools失败的解决方法(Windows、Linux、MacOs)

    前言 写这篇笔记的原因,是前几天在虚拟机 Vmware 中重新安装了几个操作系统,突然发现 VMTools 这个工具成了一个特殊的问题,以前还没有发现,因为通常它就给你自动安装了.但是大多数时候也是需 ...