zoukankan      html  css  js  c++  java
  • treap完全版模板

    这是我综合poj1442 3481 2352的treap操作 得到treap完全版模板。(经测AC)

    结构体Tree

    {

      int key; //键值

      int size; //该子树总节点个数

      int pri; //其随机值

      int son[2]; //从nocow一份代码中学来的,0表示左儿子,1表示右儿子,旋转只需一个函数即可

      (int num;) //该节点在序列中的原位置,可添加

    }

    该Treap模板支持操作:

    1. 插入值

    2. 删除值

    3. 找出第p小的节点的编号(找最大只需find(T[rt].size, rt);)

    4. 找出值小于等于key的节点个数

    (添加其他类似操作均可自己修改)

    #include <cstdio>
    #include <cstring>
    #include <ctime>
    #include <iostream>
    #include <algorithm>
    #include <cstdlib>
    #include <cmath>
    #include <utility>
    #include <vector>
    #include <queue>
    #include <map>
    #include <set>
    #define max(x,y) ((x)>(y)?(x):(y))
    #define min(x,y) ((x)>(y)?(y):(x))
    #define INF 0x3f3f3f3f
    #define MAXN 100005
    
    using namespace std;
    
    int cnt=1,rt=0; //节点编号从1开始
    
    struct Tree
    {
        int key, size, pri, son[2]; //保证父亲的pri大于儿子的pri
        void set(int x, int y, int z)
        {
            key=x;
            pri=y;
            size=z;
            son[0]=son[1]=0;
        }
    }T[MAXN];
    
    void rotate(int p, int &x)
    {
        int y=T[x].son[!p];
        T[x].size=T[x].size-T[y].size+T[T[y].son[p]].size;
        T[x].son[!p]=T[y].son[p];
        T[y].size=T[y].size-T[T[y].son[p]].size+T[x].size;
        T[y].son[p]=x;
        x=y;
    }
    
    void ins(int key, int &x)
    {
        if(x == 0)
            T[x = cnt++].set(key, rand(), 1);
        else
        {
            T[x].size++;
            int p=key < T[x].key;
            ins(key, T[x].son[!p]);
            if(T[x].pri < T[T[x].son[!p]].pri)
                rotate(p, x);
        }
    }
    
    void del(int key, int &x) //删除值为key的节点
    {
        if(T[x].key == key)
        {
            if(T[x].son[0] && T[x].son[1])
            {
                int p=T[T[x].son[0]].pri > T[T[x].son[1]].pri;
                rotate(p, x);
                del(key, T[x].son[p]);
            }
            else
            {
                if(!T[x].son[0])
                    x=T[x].son[1];
                else
                    x=T[x].son[0];
            }
        }
        else
        {
            T[x].size--;
            int p=T[x].key > key;
            del(key, T[x].son[!p]);
        }
    }
    
    int find(int p, int &x) //找出第p小的节点的编号
    {
        if(p == T[T[x].son[0]].size+1)
            return x;
        if(p > T[T[x].son[0]].size+1)
            find(p-T[T[x].son[0]].size-1, T[x].son[1]);
        else
            find(p, T[x].son[0]);
    }
    
    int find_NoLarger(int key, int &x) //找出值小于等于key的节点个数
    {
        if(x == 0)
            return 0;
        if(T[x].key <= key)
            return T[T[x].son[0]].size+1+find_NoLarger(key, T[x].son[1]);
        else
            return find_NoLarger(key, T[x].son[0]);    
    }
  • 相关阅读:
    pugixml
    C++ 头文件的理解
    图像的特征
    光圈与景深
    Unix高级环境编程
    用libtommath实现RSA算法
    【linux+C】神器 vim + 指针相关客串
    【算法25】对称子字符串的最大长度
    设计并实现同时支持多种视频格式的流媒体点播系统
    递归再一次让哥震惊了
  • 原文地址:https://www.cnblogs.com/Mathics/p/3967592.html
Copyright © 2011-2022 走看看