P2073 送花
P2073 送花
题目背景
小明准备给小红送一束花,以表达他对小红的爱意。他在花店看中了一些花,准备用它们包成花束。
题目描述
这些花都很漂亮,每朵花有一个美丽值W,价格为C。
小明一开始有一个空的花束,他不断地向里面添加花。他有以下几种操作:
操作 含义
1 W C 添加一朵美丽值为W,价格为C的花。
3 小明觉得当前花束中最便宜的一朵花太廉价,不适合送给小红,所以删除最便宜的一朵花。
2 小明觉得当前花束中最贵的一朵花太贵,他心疼自己的钱,所以删除最贵的一朵花。
-1 完成添加与删除,开始包装花束
若删除操作时没有花,则跳过删除操作。
如果加入的花朵价格已经与花束中已有花朵价格重复,则这一朵花不能加入花束。
请你帮小明写一个程序,计算出开始包装花束时,花束中所有花的美丽值的总和,以及小明需要为花束付出的总价格。
调试日志: 加粗第二处
错误日志: 加粗第一处
Solution
用这题练一下 \(Treap\)
平衡树的话就是板题了没啥好说了注意题目坑点(还有个坑是操作2、3的顺序题目给出是反过来的)
口胡一下码量少点的我的做法吧
堆 + 懒惰删除法
维护一个大根堆和一个小根堆, 堆内元素有三个值: 价钱、美丽值、 给他编的号
两个堆内元素相同, 以价钱为关键字排序, 以删除贵的为例:
开一个数组 \(ex[maxn]\) 维护这个值是否被删除过
删除堆顶, 标记其编号为被删除
下次删除小的值时取出堆顶, 若有标记则继续取出, 直到没有标记为止
这就是懒惰删除法
Code(Treap)
#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
#include<algorithm>
#include<climits>
#define LL long long
#define REP(i, x, y) for(LL i = (x);i <= (y);i++)
using namespace std;
LL RD(){
LL out = 0,flag = 1;char c = getchar();
while(c < '0' || c >'9'){if(c == '-')flag = -1;c = getchar();}
while(c >= '0' && c <= '9'){out = out * 10 + c - '0';c = getchar();}
return flag * out;
}
const LL maxn = 200019, INF = 1e9;
LL ch[maxn][2], tot, root;
LL dat[maxn], val[maxn], W[maxn];
LL sum_v[maxn], sum_w[maxn];
LL size[maxn], cnt[maxn];
LL New(LL v, LL w){
dat[++tot] = rand();
val[tot] = sum_v[tot] = v;
W[tot] = sum_w[tot] = w;
size[tot] = cnt[tot] = 1;
ch[tot][0] = ch[tot][1] = 0;
return tot;
}
void pushup(LL id){
size[id] = size[ch[id][0]] + size[ch[id][1]] + cnt[id];
sum_v[id] = sum_v[ch[id][0]] + sum_v[ch[id][1]] + val[id];
sum_w[id] = sum_w[ch[id][0]] + sum_w[ch[id][1]] + W[id];
}
void build(){
root = New(-INF, -INF);
ch[root][1] = New(INF, INF);
pushup(root);
}
void Rotate(LL &id, LL d){
LL temp = ch[id][d ^ 1];
ch[id][d ^ 1] = ch[temp][d];
ch[temp][d] = id;
id = temp;
pushup(ch[id][d]), pushup(id);
}
void insert(LL &id, LL v, LL w){
if(!id){id = New(v, w);return ;}
if(val[id] == v);
else{
LL d = v < val[id] ? 0 : 1;
insert(ch[id][d], v, w);
if(dat[id] < dat[ch[id][d]])Rotate(id, d ^ 1);
}
pushup(id);
}
void Remove(LL &id, LL v){
if(!id)return ;
if(val[id] == v){
if(cnt[id] > 1){cnt[id]--, pushup(id);return ;}
if(ch[id][0] || ch[id][1]){
if(!ch[id][1] || dat[ch[id][0]] > dat[ch[id][1]])
Rotate(id, 1), Remove(ch[id][1], v);
else Rotate(id, 0), Remove(ch[id][0], v);
pushup(id);
}
else id = 0;
return ;
}
v < val[id] ? Remove(ch[id][0], v) : Remove(ch[id][1], v);
pushup(id);
}
LL get_val_by_rank(LL id, LL rank){
if(!id)return INF;
if(rank <= size[ch[id][0]])return get_val_by_rank(ch[id][0], rank);
else if(rank <= size[ch[id][0]] + cnt[id])return val[id];
else return get_val_by_rank(ch[id][1], rank - size[ch[id][0]] - cnt[id]);
}
int main(){
build();
while(1){
LL cmd = RD();
if(cmd == 1){
LL w = RD(), v = RD();
insert(root, v, w);
}
else if(cmd == 3){
if(size[root] == 2)continue;
LL v = get_val_by_rank(root, 2);
Remove(root, v);
}
else if(cmd == 2){
if(size[root] == 2)continue;
LL v = get_val_by_rank(root, size[root] - 1);
Remove(root, v);
}
else if(cmd == -1)break;
}
printf("%lld %lld\n", sum_w[root], sum_v[root]);
return 0;
}
P2073 送花的更多相关文章
- 【洛谷】【treap/堆】P2073 送花
[题目描述:] 这些花都很漂亮,每朵花有一个美丽值W,价格为C. 小明一开始有一个空的花束,他不断地向里面添加花.他有以下几种操作: 操作 含义 1 W C 添加一朵美丽值为W,价格为C的花. 3 小 ...
- 洛谷P2073 送花 [2017年6月计划 线段树01]
P2073 送花 题目背景 小明准备给小红送一束花,以表达他对小红的爱意.他在花店看中了一些花,准备用它们包成花束. 题目描述 这些花都很漂亮,每朵花有一个美丽值W,价格为C. 小明一开始有一个空的花 ...
- [洛谷P2073] 送花
送花 题目背景 小明准备给小红送一束花,以表达他对小红的爱意.他在花店看中了一些花,准备用它们包成花束. 题目描述 这些花都很漂亮,每朵花有一个美丽值W,价格为C. 小明一开始有一个空的花束,他不断地 ...
- 洛谷 P2073 送花【Treap】题解+AC代码
题目背景 小明准备给小红送一束花,以表达他对小红的爱意.他在花店看中了一些花,准备用它们包成花束. 题目描述 这些花都很漂亮,每朵花有一个美丽值W,价格为C. 小明一开始有一个空的花束,他不断地向里面 ...
- 【题解】Luogu P2073 送花
原题传送门 这题需要用到Splay 我们用一棵splay维护金钱 考虑c<=1000000 我们珂以把每种价格现在对应的美丽值存在一个a数组中 这样讲有珂能不太清楚qaq,还是对着操作一个一个讲 ...
- luogu P2073 送花 线段树
思路&心路 一眼认定沙比提 写的比较慢,写了1小时吧 开心的交上去 卧槽,只有20? 不服不服,拿着题解的代码去对拍 Emma,<100没问题 100000数据错了,还只是错了一个数据 ...
- 洛谷 P2073 送花
这题其实可以用vector水掉! 定义: 记住要用结构体(c为价格,x为美丽值)! 以c排序. struct Node { int x,c; bool operator < (const &am ...
- Luogu P2073 送花 set
这题...一眼set...但是打了一会儿.. 记录一下每个价格对应的美丽度,顺便充当vis数组,如果美丽度不为0,说明set里已经有了... 删除好说,删*s.begin()和*--s.end()就好 ...
- Luogu P2073 送花
权值线段树的模板题 然而AC后才发现,可以用\(\tt{set}\)水过-- 权值线段树类似于用线段树来实现平衡树的一些操作,代码实现还是比较方便的 #include<iostream> ...
随机推荐
- effective c++ 笔记 (49-52)
//---------------------------15/04/27---------------------------- //#49 了解new-handler的行为 { /* 1:在o ...
- ReactJS实用技巧(1):JSX与HTML的那些不同
在项目中使用ReactJS也已经有大半年了,收获很多也踩过不少坑.不想把这个系列写成抄书似的罗列,旨在总结些常用的技巧及常见的坑,以帮助初心者快速入门,想系统学习的同学还是多阅读文档. JSX本质上与 ...
- Category Theory: 01 One Structured Family of Structures
Category Theory: 01 One Structured Family of Structures 这次看来要放弃了.看了大概三分之一.似乎不能够让注意力集中了.先更新吧. 群的定义 \( ...
- 物理机通过http访问eNSP虚拟Server
由于测试需要,本文主要通过一个简单的例子介绍通过物理机的浏览器访问华为eNSP虚拟Server,访问网页. 1.首先配置虚拟网卡的地址 2.通过华为的eNSP模拟软件,做出如下拓扑结构图,配置地址如图 ...
- Python能做什么?
Python作为一个功能强大,并且简单易学的编程语言而广受好评,那么Python都能做些什么呢?概括起来有以下几个方面: 1.Web开发: 2.大数据处理: 3.人工智能: 4.自动化运维: 5.云计 ...
- PAT甲题题解-1056. Mice and Rice (25)-模拟题
有n个老鼠,第一行给出n个老鼠的重量,第二行给出他们的顺序.1.每一轮分成若干组,每组m个老鼠,不能整除的多余的作为最后一组.2.每组重量最大的进入下一轮.让你给出每只老鼠最后的排名.很简单,用两个数 ...
- LeetCode 606. Construct String from Binary Tree根据二叉树创建字符串 (C++)
题目: You need to construct a string consists of parenthesis and integers from a binary tree with the ...
- Alpha版本项目展示要求(加入模板)
Alpha版本展示的时间暂定为11月17日课上,提前到13:00开始.如有变动,另行通知. Alpha版本项目展示要求如下: 不得使用PPT,展示所用的资料必须发表在博客上. 现场演示你们发布的软件. ...
- Daily Scrum NO.4
工作概况 符美潇(PM) 昨日完成的工作 1.Daily Scrum.日常会议及日常工作的分配和查收. 2.解决并录入了一个严重的过滤器BUG,该BUG会导致获取子链接的严重异常. 3.在TFS上进行 ...
- 20135234mqy-——信息安全系统设计基础第十四周学习总结
第九章 虚拟存储器 主要作用: 将主存看作是一个存储在磁盘上的地址空间的高速缓存,在主存中只保护活动的区域,并根据需要在磁盘和主存之间来回传送数据: 为每个进程提供了一致的地址空间,从而简化了存储器管 ...