宠物收养所 (SBT)
宠物收养所
最近,阿Q开了一间宠物收养所。收养所提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物。每个领养者都希望领养到自己满意的宠物,阿Q根据领养者的要求通过他自己发明的一个特殊的公式,得出该领养者希望领养的宠物的特点值a(a是一个正整数,a<2^31),而他也给每个处在收养所的宠物一个特点值。这样他就能够很方便的处理整个领养宠物的过程了,宠物收养所总是会有两种情况发生:被遗弃的宠物过多或者是想要收养宠物的人太多,而宠物太少。 1. 被遗弃的宠物过多时,假若到来一个领养者,这个领养者希望领养的宠物的特点值为a,那么它将会领养一只目前未被领养的宠物中特点值最接近a的一只宠物。(任何两只宠物的特点值都不可能是相同的,任何两个领养者的希望领养宠物的特点值也不可能是一样的)如果有两只满足要求的宠物,即存在两只宠物他们的特点值分别为a-b和a+b,那么领养者将会领养特点值为a-b的那只宠物。 2. 收养宠物的人过多,假若到来一只被收养的宠物,那么哪个领养者能够领养它呢?能够领养它的领养者,是那个希望被领养宠物的特点值最接近该宠物特点值的领养者,如果该宠物的特点值为a,存在两个领养者他们希望领养宠物的特点值分别为a-b和a+b,那么特点值为a-b的那个领养者将成功领养该宠物。 一个领养者领养了一个特点值为a的宠物,而它本身希望领养的宠物的特点值为b,那么这个领养者的不满意程度为abs(a-b)。【任务描述】你得到了一年当中,领养者和被收养宠物到来收养所的情况,希望你计算所有收养了宠物的领养者的不满意程度的总和。这一年初始时,收养所里面既没有宠物,也没有领养者。
Input
第一行为一个正整数n,n<=80000,表示一年当中来到收养所的宠物和领养者的总数。接下来的n行,按到来时间的先后顺序描述了一年当中来到收养所的宠物和领养者的情况。每行有两个正整数a, b,其中a=0表示宠物,a=1表示领养者,b表示宠物的特点值或是领养者希望领养宠物的特点值。(同一时间呆在收养所中的,要么全是宠物,要么全是领养者,这些宠物和领养者的个数不会超过10000个)
Output
仅有一个正整数,表示一年当中所有收养了宠物的领养者的不满意程度的总和mod 1000000以后的结果。
Sample Input
5
0 2
0 4
1 3
1 2
1 5
Sample Output
3
(abs(3-2) + abs(2-4)=3,最后一个领养者没有宠物可以领养)
【分析】直接套模板。当宠物剩余,平衡树里存宠物的特点值;当购买者剩余,平衡树里存购买者的喜爱值。
#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <string>
#include <map>
#include <stack>
#include <queue>
#include <vector>
#define inf 100000000
#define met(a,b) memset(a,b,sizeof a)
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
typedef long long ll;
using namespace std;
const int N =1e6+;
const int M = 4e5+;
struct SBT {
int left,right,size,key;
void Init() {
left=right=;
size=;
}
} tree[N];
int tot,root;
void left_rotate(int &x) { //左旋
int y=tree[x].right;
tree[x].right=tree[y].left;
tree[y].left=x;
tree[y].size=tree[x].size;
tree[x].size=tree[tree[x].left].size+tree[tree[x].right].size+;
x=y;
}
void right_rotate(int &x) { //右旋
int y=tree[x].left;
tree[x].left=tree[y].right;
tree[y].right=x;
tree[y].size=tree[x].size;
tree[x].size=tree[tree[x].left].size+tree[tree[x].right].size+;
x=y;
}
void maintain(int &x,int flag) {
if(flag==) {
if(tree[tree[tree[x].left].left].size > tree[tree[x].right].size)
right_rotate(x);
else if(tree[tree[tree[x].left].right].size > tree[tree[x].right].size)
left_rotate(tree[x].left),right_rotate(x);
else return;
} else {
if(tree[tree[tree[x].right].right].size > tree[tree[x].left].size)
left_rotate(x);
else if(tree[tree[tree[x].right].left].size > tree[tree[x].left].size)
right_rotate(tree[x].right),left_rotate(x);
else return;
}
maintain(tree[x].left,);
maintain(tree[x].right,);
maintain(x,);
maintain(x,);
}
//插入元素,相同元素放在右子树中
void insert(int &x,int key) {
if(x==) {
x=++tot;
tree[x].Init();
tree[x].key=key;
} else {
tree[x].size++;
if(key<tree[x].key)insert(tree[x].left,key);
else insert(tree[x].right,key);
maintain(x,key>=tree[x].key);
}
}
//删除key值的元素
int del(int &x,int key) {
if(!x)return ;
tree[x].size--;
if(key==tree[x].key || (key<tree[x].key&&tree[x].left==)
|| (key>tree[x].key&&tree[x].right==)) {
if(tree[x].left && tree[x].right) {
int p=del(tree[x].left,key+);
tree[x].key=tree[p].key;
return p;
} else {
int p=x;
x=tree[x].left+tree[x].right;
return p;
}
} else return del(key<tree[x].key?tree[x].left:tree[x].right,key);
}
//**************
//得到前驱,小于
//返回前驱的节点编号
//**************
int get_pre(int &r,int y,int k) {
if(r==)return y;
if(k>tree[r].key)//加个等号就是小于等于
return get_pre(tree[r].right,r,k);
else return get_pre(tree[r].left,y,k);
} //**************
//得到后继,大于
//返回后继的节点编号
//*********
int get_next(int &r,int y,int k) {
if(r==)return y;
if(k<tree[r].key)//加个等号就是大于等于
return get_next(tree[r].left,r,k);
else return get_next(tree[r].right,y,k);
}
//**************
//查找是否存在key值为k的元素
//不存在返回0,存在返回节点编号
//***************
int find(int t,int k) {
while(t && k!=tree[t].key)
t=k<tree[t].key?find(tree[t].left,k):find(tree[t].right,k);
return t;
}
//调试,按顺序输出
void inorder(int &x) {
if(x==)return;
else {
inorder(tree[x].left);
cout<<" "<<tree[x].key<<" "<<tree[x].size<<endl;
inorder(tree[x].right);
}
} //***********
//得到前驱,返回的是小于v的key值,不存在v
//**********
int Pred(int t,int v) {
if(!t)return v;
if(v<=tree[t].key)return Pred(tree[t].left,v);
else {
int tmp=Pred(tree[t].right,v);
return v==tmp?tree[t].key:tmp;
}
}
//***********
//得到后继,返回的是大于v的key值,不存在v
//**********
int Succ(int t,int v) {
if(!t)return v;
if(v>=tree[t].key)return Succ(tree[t].right,v);
else {
int tmp=Succ(tree[t].left,v);
return v==tmp?tree[t].key:tmp;
}
} //得到第k大的元素
int get_Kth(int &x,int k) {
if(k<=tree[tree[x].left].size)
return get_Kth(tree[x].left,k);
if(k>tree[tree[x].left].size+)
return get_Kth(tree[x].right,k-tree[tree[x].left].size-);
return tree[x].key;
} //取最大值
//返回最大值的节点编号
int Get_Max(int x) {
while(tree[x].right)x=tree[x].right;
return x;
}
//取最小值
//返回最小值的节点编号
int Get_Min(int x) {
while(tree[x].left)x=tree[x].left;
return x;
} const int MOD=;
int main() {
int n,a,b;
while(scanf("%d",&n)==) {
root=tot=;//这个初始化一定不要忘记
int flag;
int ans=;
while(n--) {
scanf("%d%d",&a,&b);
if(root==) { //空的
flag=a;
insert(root,b);
} else {
if(a==flag) {
insert(root,b);
} else {
if(find(root,b)) {
del(root,b);
continue;
}
int t1=get_pre(root,,b);
int t2=get_next(root,,b);
if(t1==) {
ans+=abs(tree[t2].key-b);
del(root,tree[t2].key);
ans%=MOD;
} else if(t2==) {
ans+=abs(b-tree[t1].key);
del(root,tree[t1].key);
ans%=MOD;
} else {
if(abs(b-tree[t1].key)<=abs(tree[t2].key-b) ) {
ans+=abs(b-tree[t1].key);
del(root,tree[t1].key);
ans%=MOD;
} else {
ans+=abs(tree[t2].key-b);
del(root,tree[t2].key);
ans%=MOD;
}
}
}
}
}
printf("%d\n",ans);
}
return ;
}
宠物收养所 (SBT)的更多相关文章
- Bzoj1208 [HNOI2004]宠物收养所
Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 7457 Solved: 2960 Description 最近,阿Q开了一间宠物收养所.收养所提供两 ...
- BZOJ 1208: [HNOI2004]宠物收养所
1208: [HNOI2004]宠物收养所 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 7514 Solved: 2982[Submit][Sta ...
- 宠物收养所(bzoj1208)
Description 最近,阿Q开了一间宠物收养所.收养所提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物.每个领养者都希望领养到自己满意的宠物,阿Q根据领养者的要求通过他自己发明的一个特 ...
- 【BZOJ1208】[HNOI2004]宠物收养所 Splay
还是模板题,两颗splay,找点删即可. #include <iostream> #include <cstdio> #include <cstdlib> #def ...
- 【BZOJ-1208】宠物收养所 Splay
1208: [HNOI2004]宠物收养所 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 6638 Solved: 2601[Submit][Sta ...
- BZOJ1208 宠物收养所
Description 最近,阿Q开了一间宠物收养所.收养所提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物.每个领养者都希望领养到自己满意的宠物,阿Q根据领养者的要求通过他自己发明的一个特 ...
- C++之路进阶——codevs1285(宠物收养所)
1285 宠物收养所 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 最近,阿Q开了一间宠物收养所.收养所提供两种服 ...
- bzoj 1208: [HNOI2004]宠物收养所 set
1208: [HNOI2004]宠物收养所 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 7328 Solved: 2892[Submit][Sta ...
- BZOJ_1208_&_Codevs_1258_[HNOI2004]_宠物收养所_(平衡树/set)
描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1208 (据说codevs要更新?就不放codevs的地址了吧...) 有宠物和人,每个单位都有 ...
随机推荐
- 【Gradient Boosted Decision Tree】林轩田机器学习技术
GBDT之前实习的时候就听说应用很广,现在终于有机会系统的了解一下. 首先对比上节课讲的Random Forest模型,引出AdaBoost-DTree(D) AdaBoost-DTree可以类比Ad ...
- chromedriver版本支持的Chrome版本
下载chromedriver,链接:http://chromedriver.storage.googleapis.com/index.html chromedirver版本 支持的Chrome版本 ...
- 洛谷P1071潜伏者(提高组)
题目描述 R国和S国正陷入战火之中,双方都互派间谍,潜入对方内部,伺机行动.历尽艰险后,潜伏于S国的R 国间谍小C终于摸清了 S 国军用密码的编码规则: 1. S国军方内部欲发送的原信息经过加密后在网 ...
- 没有外网情况下linux安装mysql
首先linux要使用局域网 环境要求:局域网, windows系统, linux系统, mysql安装包mysql.tar.gz 注意:32位操作系统用32位安装包,64位系统用64位安装包,不 ...
- 【bzoj4870】[Shoi2017]组合数问题 dp+快速幂/矩阵乘法
题目描述 输入 第一行有四个整数 n, p, k, r,所有整数含义见问题描述. 1 ≤ n ≤ 10^9, 0 ≤ r < k ≤ 50, 2 ≤ p ≤ 2^30 − 1 输出 一行一个整数 ...
- [AGC010E] Rearranging [拓扑排序+堆]
题面 传送门 思路 首先,一个显然的结论是:Alice调整过后的序列中任意两个不互质的数的相对顺序无法改变 那么我们可以以这个性质为突破口 我们在两个不互质的权值的点之间连一条边(没错这是个图论题!! ...
- [poj] 3057 Evacuation
原题 题目大意 墙壁"X",空区域(都是人)".", 门"D". 人向门移动通过时视为逃脱,门每秒能出去一个人,人可以上下左右移动,墙阻止移 ...
- Linux系统—— core 文件
Linux core 文件介绍 1. core文件的简单介绍 在一个程序崩溃时,它一般会在指定目录下生成一个core文件.core文件仅仅是一个内存映象(同时加上调试信息),主要是用来调试的. 2. ...
- Hibernate中多对多的annotation的写法(中间表可以有多个字段)
2011-07-04 6:52 一般情况下,多对多的关联关系是需要中间表的: 情况一:如果中间表仅仅是做关联用的,它里面仅有2个外键做联合主键,则使用ManyToMany(不用写中间表的Model,只 ...
- Yii2.0 使用createcommand从数据库查询出来的int类型变成了string型
在给客户端写接口文档时,会给每个返回的字段标上数据类型,客户端会根据标注的类型来解析数据,如果标注的类型错误,会导致客户端解析出错,造成崩溃. 一直以来我都是用的Yii进行项目的开发,之前使用Yii1 ...