leecode 1206|跳表的设计

跳表

跳表,一种链表数据结构,其增删改茶的效率能和平衡树相媲美

leecode1206

可以看上面的那个动画,动画效果很贴切。

我简单讲讲它的机制吧,每个节点不单单是一个,测试好几层,然后同一层的节点和统一节点的next 采用单链表产生联系

最核心的东西在于find

这也是为什么单链表的增删改查,花费开销最多的地方。

那它是怎么查的?

我们已经知道了跳表的结构了,最底层肯定是最全的,如果暴力最底层,那就和单链表没什么区别了。

它的这个查恰巧结合了跳表的结构,如果在上面某个层找到了节点x 的pre那么接下来的操作都好办了。

那它是怎么找到节点x 的pre呢?

从最上层,依次找该节点的next 知道找到pre 如果在这一层还没找到(因为最后一个节点的next 为nullptr),那么会在这个节点的下一层继续找同层找next。依此类推,最后肯定能找到的。

cpp 复制代码
class Skiplist {
private:
    static const int MAX_LEVEL = 10;
    struct Node{
        int val;
        std::vector<Node*> ne;
        Node(int _val, int level):val(_val), ne(level, nullptr){};
    };
    int level;
    Node* head;
    float probability;

    int randomLevel(){
        int lvl = 1;
        while((float) std::rand() / RAND_MAX < probability && lvl < MAX_LEVEL){
            lvl++;
        }
        return lvl;
    }

    void find(int t, std::vector<Node*>& ns){
        Node* cur = head;
        for(int i = level - 1; i >= 0; i--){
            while(cur->ne[i] != nullptr && cur->ne[i]->val < t){
                cur = cur->ne[i];
            }
            ns[i] = cur;
        }
    }
public:
    Skiplist() :level(MAX_LEVEL), head(new Node(-1, MAX_LEVEL)), probability(0.5){
        std::srand(std::time(0));

    }
    
    bool search(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);
        Node* target = ns[0]->ne[0];
        return target != nullptr && target->val == t;
    }
    
    void add(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);

        int nodeLevel = randomLevel();
        Node* newNode = new Node(t, nodeLevel);
        for(int i = 0; i < nodeLevel; i++){
            newNode->ne[i] = ns[i]->ne[i];
            ns[i]->ne[i] = newNode;
        }
    }
    
    bool erase(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);
        Node* target = ns[0]->ne[0];
        if(target == nullptr || target->val != t){
            return false;
        }

        for(int i = 0; i < level && ns[i]->ne[i] == target; i++){
            ns[i]->ne[i] = ns[i]->ne[i]->ne[i];
        }
        delete target;
        return true;

    }

    ~Skiplist(){
        Node* cur = head;
        while(cur){
            Node* next = cur->ne[0];
            delete cur;
            cur = next;
        }
    }
};

/**
 * Your Skiplist object will be instantiated and called as such:
 * Skiplist* obj = new Skiplist();
 * bool param_1 = obj->search(target);
 * obj->add(num);
 * bool param_3 = obj->erase(num);
 */

再贴一份,

cpp 复制代码
#include <iostream>
#include <vector>
#include <cstdlib>
#include <ctime>

class Skiplist {
private:
    static const int MAX_LEVEL = 10;
    struct Node {
        int val;
        std::vector<Node*> ne;
        Node(int _val, int level) : val(_val), ne(level, nullptr) {}
    };

    int level;
    Node* head;
    float probability;

    int randomLevel() {
        int lvl = 1;
        while ((float)std::rand() / RAND_MAX < probability && lvl < MAX_LEVEL) {
            lvl++;
        }
        return lvl;
    }

    void find(int t, std::vector<Node*>& ns) {
        Node* cur = head;
        for (int i = level - 1; i >= 0; i--) {
            while (cur->ne[i] != nullptr && cur->ne[i]->val < t) {
                cur = cur->ne[i];
            }
            ns[i] = cur;
        }
    }

public:
    Skiplist() : level(MAX_LEVEL), head(new Node(-1, MAX_LEVEL)), probability(0.5) {
        std::srand(std::time(0));
    }

    bool search(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);
        Node* target = ns[0]->ne[0];
        return target != nullptr && target->val == t;
    }

    void add(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);
        int nodeLevel = randomLevel();
        Node* newNode = new Node(t, nodeLevel);
        for (int i = 0; i < nodeLevel; i++) {
            newNode->ne[i] = ns[i]->ne[i];
            ns[i]->ne[i] = newNode;
        }
    }

    bool erase(int t) {
        std::vector<Node*> ns(level, nullptr);
        find(t, ns);
        Node* target = ns[0]->ne[0];
        if (target == nullptr || target->val != t) {
            return false;
        }
        for (int i = 0; i < level && ns[i]->ne[i] == target; i++) {
            ns[i]->ne[i] = ns[i]->ne[i]->ne[i];
        }
        delete target;
        return true;
    }

    ~Skiplist() {
        Node* cur = head;
        while (cur) {
            Node* next = cur->ne[0];
            delete cur;
            cur = next;
        }
    }
};

int main() {
    Skiplist skiplist;
    skiplist.add(1);
    skiplist.add(2);
    skiplist.add(3);
    std::cout << skiplist.search(1) << std::endl;  // returns true
    std::cout << skiplist.search(4) << std::endl;  // returns false
    skiplist.add(4);
    std::cout << skiplist.search(4) << std::endl;  // returns true
    std::cout << skiplist.erase(4) << std::endl;   // returns true
    std::cout << skiplist.search(4) << std::endl;  // returns false
    return 0;
}

redis

参考

相关推荐
丛雨要玩游戏15 分钟前
字符函数和字符串函数
c语言·开发语言·算法
ad钙奶长高高36 分钟前
【C语言】初始C语言
c语言·开发语言·算法
侯小啾41 分钟前
【17】C语言-gets() 与 fgets() 函数
c语言·开发语言
是苏浙2 小时前
零基础入门C语言之枚举和联合体
c语言·开发语言
ACP广源盛139246256734 小时前
(ACP广源盛)GSV2231---DisplayPort 1.4 MST 到 HDMI 2.0/DP/Type-C 转换器(带嵌入式 MCU)
c语言·开发语言·单片机·嵌入式硬件·音视频·mst
QT 小鲜肉4 小时前
【QT/C++】Qt网络编程进阶:UDP通信和HTTP请求的基本原理和实际应用(超详细)
c语言·网络·c++·笔记·qt·http·udp
Yurko135 小时前
【C语言】选择结构和循环结构的进阶
c语言·开发语言·学习
范纹杉想快点毕业5 小时前
12个月嵌入式进阶计划ZYNQ 系列芯片嵌入式与硬件系统知识学习全计划(基于国内视频资源)
c语言·arm开发·单片机·嵌入式硬件·学习·fpga开发·音视频
木木木丫6 小时前
嵌入式项目:韦东山驱动开发第六篇 项目总结——显示系统(framebuffer编程)
c语言·c++·驱动开发·dsp开发
柳鲲鹏6 小时前
RGB转换为NV12,查表式算法
linux·c语言·算法