BZOJ 1208 (可持久化Treap,合并与分裂操作)

2024-08-23 14:38

本文主要是介绍BZOJ 1208 (可持久化Treap,合并与分裂操作),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1208: [HNOI2004]宠物收养所

Time Limit: 10 Sec   Memory Limit: 162 MB
Submit: 4965   Solved: 1902
[ Submit][ Status][ Discuss]

Description

最近,阿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,最后一个领养者没有宠物可以领养)
/*
Treap的简单应用 不过需要注意几个细节! 寻找前驱和后继节点的时候,是找的value值为v的严格前驱和非严格后继节点;
*/
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std;
#define inf 0x0f0f0f0f
#define mod 1000000
int abs(int x){ return (x>=0?x:-x); }
struct Node {Node *ch[2];int rank,value,siz;Node(){ ch[0] = ch[1] = NULL; }Node(int v,int r) {ch[0] = ch[1] = NULL;value = v; rank = r;siz = 1;}void push_up() {siz = 1;if(ch[0]!=NULL) siz += ch[0]->siz;if(ch[1]!=NULL) siz += ch[1]->siz;}
}*root;
int res,tag;
typedef pair<Node *,Node *> DNode;
Node *merge(Node *(&node1),Node *(&node2)) {if(node1==NULL) return node2;if(node2==NULL) return node1;if(node1->rank > node2->rank){node1->ch[1] = merge(node1->ch[1],node2);node1->push_up();return node1;}else{node2->ch[0] = merge(node1,node2->ch[0]);node2->push_up();return node2;}
}
DNode split(Node *(&node),int k) {if(node==NULL) return DNode(NULL,NULL);if(k<=0)         return DNode(NULL,node);if(node->siz<=k) return DNode(node,NULL);DNode ans = DNode(NULL,NULL);if(node->ch[0]!=NULL&&node->ch[0]->siz >= k) {ans = split(node->ch[0],k);node->ch[0] = ans.second;node->push_up();ans.second = node;}else {int size = (node->ch[0]!=NULL?node->ch[0]->siz+1:1);ans = split(node->ch[1],k-size);node->ch[1] = ans.first;node->push_up();ans.first = node;}return ans;
}
//找node的严格前驱节点
Node *getPre(Node *(&node),int v) {if(node == NULL) return NULL;Node *ans = node;if(node->value >= v) ans = getPre(node->ch[0],v);else {ans = getPre(node->ch[1],v);if(ans==NULL) ans = node;}return ans;
}
//找node的非严格后继节点
Node *getNext(Node *(&node),int v) {if(node == NULL) return NULL;Node *ans = node;if(node->value >= v){ans = getNext(node->ch[0],v);if(ans==NULL) ans = node;}else ans = getNext(node->ch[1],v);return ans;
}
//查找严格<v节点个数
int search(Node *(&node),int v) {if(node == NULL) return 0;if(node->value >= v) return search(node->ch[0],v);else {int k = 1;if(node->ch[0]!=NULL) k += node->ch[0]->siz;return k + search(node->ch[1],v);}
}
void insert(int v) {int k = search(root,v);DNode tem = split(root,k);Node *node = new Node(v,rand());tem.first = merge(tem.first,node);root = merge(tem.first,tem.second);
}
void earse(int v){Node *left = getPre(root,v);Node *right = getNext(root,v);int k = search(root,v),ldis = inf,rdis = inf;if(left!=NULL)  ldis = v - left->value;if(right!=NULL) rdis = right->value - v;//printf("%d,%d\n",ldis,rdis);if(ldis<=rdis){res = (res + ldis%mod)%mod;DNode tem1 = split(root,k-1);DNode tem2 = split(tem1.second,1);delete tem2.first; tem2.first = NULL;root = merge(tem1.first,tem2.second);}else {res = (res + rdis%mod)%mod;DNode tem1 = split(root,k);DNode tem2 = split(tem1.second,1);delete tem2.first; tem2.first = NULL;root = merge(tem1.first,tem2.second);}
}
void solve(int flag,int v){if(root==NULL) tag = flag;if(flag==tag) insert(v);else          earse(v);
}
void delet(Node *(node)){if(node==NULL) return;delet(node->ch[0]);if(node->ch[0]!=NULL){delete node->ch[0];node->ch[0]=NULL;}delet(node->ch[1]);if(node->ch[1]!=NULL){delete node->ch[1];node->ch[1]=NULL;}delete node; node = NULL;
}
void init(){res = tag = 0;delet(root);root = NULL;
}
int main() {int n,flag,v;//freopen("Test.txt","r",stdin);while(~scanf("%d",&n)){init();for(int i = 1;i<=n;i++){scanf("%d%d",&flag,&v);solve(flag,v);}printf("%d\n",res);}return 0;
}






这篇关于BZOJ 1208 (可持久化Treap,合并与分裂操作)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1099669

相关文章

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

day-51 合并零之间的节点

思路 直接遍历链表即可,遇到val=0跳过,val非零则加在一起,最后返回即可 解题过程 返回链表可以有头结点,方便插入,返回head.next Code /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode() {}*

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

动手学深度学习【数据操作+数据预处理】

import osos.makedirs(os.path.join('.', 'data'), exist_ok=True)data_file = os.path.join('.', 'data', 'house_tiny.csv')with open(data_file, 'w') as f:f.write('NumRooms,Alley,Price\n') # 列名f.write('NA

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

Java IO 操作——个人理解

之前一直Java的IO操作一知半解。今天看到一个便文章觉得很有道理( 原文章),记录一下。 首先,理解Java的IO操作到底操作的什么内容,过程又是怎么样子。          数据来源的操作: 来源有文件,网络数据。使用File类和Sockets等。这里操作的是数据本身,1,0结构。    File file = new File("path");   字

MySQL——表操作

目录 一、创建表 二、查看表 2.1 查看表中某成员的数据 2.2 查看整个表中的表成员 2.3 查看创建表时的句柄 三、修改表 alter 3.1 重命名 rename 3.2 新增一列 add 3.3 更改列属性 modify 3.4 更改列名称 change 3.5 删除某列 上一篇博客介绍了库的操作,接下来来看一下表的相关操作。 一、创建表 create

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

PHP7扩展开发之流操作

前言 啥是流操作?简单来讲就是对一些文件,网络的IO操作。PHP已经把这些IO操作,封装成流操作。这节,我们将使用PHP扩展实现一个目录遍历的功能。PHP示例代码如下: <?phpfunction list_dir($dir) {if (is_dir($dir) === false) {return;} $dh = opendir($dir);if ($dh == false) {ret

浙大数据结构:树的定义与操作

四种遍历 #include<iostream>#include<queue>using namespace std;typedef struct treenode *BinTree;typedef BinTree position;typedef int ElementType;struct treenode{ElementType data;BinTree left;BinTre