【传送门】

前言

这一篇题解并不是为了讲什么算法,只是总结一下平衡树在OI考试中的注意事项。

题意简化(给不想看题目的小伙伴们一点福利)

给你两堆数,每一次给你一个数,每一次在另外一堆数中找到这个数的前缀后继,删去前驱后继中较靠近的,得到了分数为两个数的差的绝对值。请你让这个分数最小。
(题意简化的应该不能在简化了吧qwq)


做法1&&2&&3(非正常做法)

我们都知道,所有的平衡树的题目都可以用不定长数组vector和不允许重复元素的set或者是允许重复元素的muliset来实现。
这个东西在考场上是救命的,如果你剩下的时间不多了,那么是在不行就用以上的办法来偏分。
分析一下vectorsetmuliset的优缺点。
不定长数组vector实现起来非常的简单,只要你熟悉指针和vector的正常操作。
但是vector很容易溢出,长度一大,很有可能就RE,那个时候想哭都没时间哭了。

#include <bits/stdc++.h>
using namespace std;
int n;
vector<int>v;
int main() {
    scanf("%d",&n);
    for(int i=1;i<=n;i++) {
        int op=0,x=0;
        scanf("%d%d",&op,&x);
        if(op==1) v.insert(upper_bound(v.begin(),v.end(),x),x);
        else if(op==2) v.erase(lower_bound(v.begin(),v.end(),x));
        else if(op==3) printf("%d\n",lower_bound(v.begin(),v.end(),x)-v.begin()+1);
        else if(op==4) printf("%d\n",v[x-1]);
        else if(op==5) printf("%d\n",*--lower_bound(v.begin(),v.end(),x));
        else if(op==6) printf("%d\n",*upper_bound(v.begin(),v.end(),x));
    }
    return 0;
}

实现非常的简单,但是需要注意指针的变换。
但是这一道题目我们用vector,蒟蒻我用了差不多5s才跑出大的数据。(数据从LOJ上来的)
所以setmuliset可以更加快速的完成我们的任务。
set差别并不是太大,一个可以实现没有重复的,一个实现有重复的。
来自chhokmah的实测,set比muliset要快。
所以总结一下:

  • vector好写,但是容易错,不推荐使用。
  • set可以实现无重复,推荐使用。
  • muliset可以实现有重复,推荐使用。

做法4正常的平衡树

这里以treap为例。
简单介绍一下treap,treap是tree和heap的结合,每一次我们需要用自己给节点附加的rd值来维护平衡树的平衡性。
代码实现非常的简单,但是我打的比较冗长。
那么再回到这一道题目,如果我们只需要建立两个平衡树。(其实一棵就足够了,因为一棵有节点的时候另外一棵一定是没有节点的。)
每一次找前驱和后继,判断更加接近的那一个,更新答案,并删除前驱或者是后继。
如果树为空或者是己方的树有节点,那么就直接插入。

#include <bits/stdc++.h>
#define N 80005
#define mod 1000000
#define ll long long
#define inf 0x3f3f3f3f
using namespace std;
template <typename T>
inline void read(T &x) {
    x = 0; T fl = 1; char ch = 0;
    for (; ch < '0' || ch > '9'; ch = getchar())
        if (ch == '-') fl = -1;
    for (; ch >= '0' && ch <= '9'; ch = getchar())
        x = (x << 1) + (x << 3) + (ch ^ 48);
    x *= fl;
}
struct Treap {
    int tot, rt;
    struct Node {
        int cnt, sz, rd, ch[2], val;
        void Init (int Val) { val = Val; ch[0] = ch[1] = 0; cnt = sz = 1; rd = rand() % 100; }
    } tr[N];
    Treap() { tot = 0; memset(tr, 0, sizeof(tr)); }
    void pushup (int nod) { tr[nod].sz = tr[tr[nod].ch[0]].sz + tr[tr[nod].ch[1]].sz + tr[nod].cnt; }
    void rotate(int &nod, int d) {
        int k = tr[nod].ch[d ^ 1]; tr[nod].ch[d ^ 1] = tr[k].ch[d]; tr[k].ch[d] = nod;
        pushup(nod); pushup(k); nod = k;
    }
    void ins(int &nod, int val) {
        if (!nod) nod = ++ tot, tr[nod].Init(val);
        else if (val == tr[nod].val) tr[nod].sz ++, tr[nod].cnt ++;
        else {
            int d = (val > tr[nod].val);
            ins(tr[nod].ch[d], val);
            if (tr[nod].rd < tr[tr[nod].ch[d]].rd) rotate(nod, d ^ 1);
            pushup(nod);
        }
    }
    void del(int &nod, int val) {
        if (!nod) return;
        if (val < tr[nod].val) del(tr[nod].ch[0], val);
        else if (val > tr[nod].val) del(tr[nod].ch[1], val);
        else {
            if (!tr[nod].ch[0] && !tr[nod].ch[1]) {tr[nod].sz --, tr[nod].cnt --; if (tr[nod].cnt == 0) nod = 0;}
            else if (tr[nod].ch[0] && !tr[nod].ch[1]) { rotate(nod, 1) ; del(tr[nod].ch[1], val);}
            else if (!tr[nod].ch[0] && tr[nod].ch[1]) { rotate(nod, 0); del(tr[nod].ch[0], val); }
            else {
                int d = tr[tr[nod].ch[0]].rd > tr[tr[nod].ch[1]].rd;
                rotate(nod, d); del(tr[nod].ch[d], val);
            }
        }
    }
    int pre(int nod, int val) {
        if (!nod) return -inf;
        if (tr[nod].val > val) return pre(tr[nod].ch[0], val);
        else return max(tr[nod].val , pre(tr[nod].ch[1], val));
    }
    int suc(int nod, int val) {
        if (!nod ) return inf;
        if (tr[nod].val < val) return suc(tr[nod].ch[1], val);
        else return min(tr[nod].val, suc(tr[nod].ch[0], val));
    }
}Pet, Cus;
int n;
ll ans = 0ll;
int main () {
    srand(19260817);
    read(n);
    while (n --) {
        int opt, x; read(opt); read(x);
        if (opt == 0) { // Pet
            if (Cus.rt == 0) Pet.ins(Pet.rt, x);
            else {
                int lst = Cus.pre(Cus.rt, x), nxt = Cus.suc(Cus.rt, x);
                if (abs(x - lst) <= abs(nxt - x))  { Cus.del(Cus.rt, lst); ans = (ans + 1ll * abs(x - lst)) % mod; }
                else { Cus.del(Cus.rt, nxt); ans = (ans + 1ll * abs(nxt - x)) % mod;}
            }
        }
        else { // Customer
            if (Pet.rt == 0) Cus.ins(Cus.rt, x);
            else {
                int lst = Pet.pre(Pet.rt, x), nxt = Pet.suc(Pet.rt, x);
                if (abs(x - lst) <= abs(nxt - x))  { Pet.del(Pet.rt, lst); ans = (ans + 1ll * abs(x - lst)) % mod; }
                else { Pet.del(Pet.rt, nxt); ans = (ans + 1ll * abs(nxt - x)) % mod;}
            }
        }
    }
    printf("%lld\n", ans % mod);
    return 0;
}

[luogu2286][HNOI2004]宠物收养场【平衡树】的更多相关文章

  1. [HNOI2004]宠物收养场 Treap前驱后继

    凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领养者希望领 ...

  2. BZOJ1208[HNOI2004]宠物收养场——treap

    凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领养者希望领 ...

  3. 洛谷 P2286 [HNOI2004]宠物收养场

    题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领 ...

  4. LG_2286_[HNOI2004]宠物收养场

    题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领 ...

  5. P2286 [HNOI2004]宠物收养场

    题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领 ...

  6. 洛谷P2286 [HNOI2004]宠物收养场【Treap】题解+AC代码

    题目传送门啦~啦~啦~ 题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的 ...

  7. 洛谷P2286 [HNOI2004]宠物收养场

    题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领 ...

  8. [HNOI2004]宠物收养场 BZOJ1208 splay tree

    题目描述 凡凡开了一间宠物收养场.收养场提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物. 每个领养者都希望领养到自己满意的宠物,凡凡根据领养者的要求通过他自己发明的一个特殊的公式,得出该领 ...

  9. 1208. [HNOI2004]宠物收养场【平衡树-splay】

    Description 最近,阿Q开了一间宠物收养所.收养所提供两种服务:收养被主人遗弃的宠物和让新的主人领养这些宠物.每个领养者都希望领养到自己满意的宠物,阿Q根据领养者的要求通过他自己发明的一个特 ...

随机推荐

  1. 远程调用HBase出错,尝试10次后,报org.apache.hadoop.hbase.MasterNotRunningException错误

    网上的解决方案挺多的,但都不适用于我今天下午碰到的情况. 环 境:HBase-0.90.3在debian 6下,客户端在windows上.我用之前的HBase服务器是没问题的,但重新解压并配置后就有问 ...

  2. echarts使用笔记四:双Y轴

    1.双Y轴显示数量和占比 app.title = '坐标轴刻度与标签对齐'; option = { title : { //标题 x : 'center', y : 5, text : '数量和占比图 ...

  3. Nginx三部曲(2)性能

    我们会告诉你 Nginx 如何工作及其背后的理念,还有如何优化以加快应用的性能,如何安装启动和保持运行. 这个教程有三个部分: 基本概念 —— 这部分需要去了解 Nginx 的一些指令和使用场景,继承 ...

  4. ORA-12541:TNS:无监听程序 解决办法

    昨天我在一台win7笔记本中安装了oracle11g,然后打算用另一个win10的笔记本使用plsql developer局域网内连接访问oracle数据库.但是遇到ORA-12541:TNS:无监听 ...

  5. 反射获取Class对象

    实际演示

  6. Java8 Hash改进/内存改进

    又开新坑o(*≧▽≦)ツ讲讲几个Java版本的特性,先开始Java8, HashMap的改进 HashMap采用哈希算法,先使用hashCode()判断哈希值是否相同,如果相同,再使用equals() ...

  7. Spring Aop实例@Aspect、@Before、@AfterReturning@Around 注解方式配置

    用过spring框架进行开发的人,多多少少会使用过它的AOP功能,都知道有@Before.@Around和@After等advice.最近,为了实现项目中的输出日志和权限控制这两个需求,我也使用到了A ...

  8. Python 版百度站长平台链接主动推送脚本

    如果自己的网站需要被百度收录,可以在搜索结果中找到,就需要将网站的链接提交给百度.依靠百度的爬虫可能无法检索到网站所有的内容,因此可以主动将链接提交给百度. 在百度的站长平台上介绍了链接提交方法,目前 ...

  9. Java多线程6:Synchronized锁代码块(this和任意对象)

    一.Synchronized(this)锁代码块 用关键字synchronized修饰方法在有些情况下是有弊端的,若是执行该方法所需的时间比较长,线程1执行该方法的时候,线程2就必须等待.这种情况下就 ...

  10. 和docket的第一次亲密接触

    很久很久以前,第一次听说docker时,感觉很高大上,同时自我感觉会很难.所以一直没有详细了解.前一段时间偶尔看到关于docker的详细介绍,于是乎来了兴趣.自已折腾了一下,发现不是想象中的那么难. ...