Redis(128)Redis的跳表(Skip List)是如何实现的?

在 Redis 中,跳表(Skip List)是一种用于实现有序集合(Sorted Set)底层数据结构的算法。与传统的平衡树(如红黑树)相比,跳表结构相对简单,并且在实现上更加直观。

跳表通过在有序链表上构建多级索引来加速搜索操作,跳表的每一层都是一个有序链表,且每一层的元素是下一层的子集,通过多级索引,可以在对数级别的时间复杂度内进行插入、删除和查找操作。

跳表的基本结构

一个跳表节点包含以下几个部分:

  • score:节点的分数,用于排序。
  • value:节点的值。
  • forward:一个数组,包含指向不同层级的下一个节点的指针。
  • level:节点的层级。

跳表的具体实现

下面是跳表的基本实现,包括节点的定义、跳表的初始化、插入、删除和查找操作。

节点定义

首先,定义跳表节点结构。

java 复制代码
class SkipListNode {
    double score;
    String value;
    SkipListNode[] forward;

    public SkipListNode(int level, double score, String value) {
        this.score = score;
        this.value = value;
        this.forward = new SkipListNode[level + 1];
    }
}
跳表定义

定义跳表结构。

java 复制代码
class SkipList {
    private static final int MAX_LEVEL = 16;
    private static final double P = 0.5;
    private SkipListNode head;
    private int level;

    public SkipList() {
        this.level = 0;
        this.head = new SkipListNode(MAX_LEVEL, 0, null);
    }

    // 随机生成节点层级
    private int randomLevel() {
        int lvl = 0;
        while (Math.random() < P && lvl < MAX_LEVEL) {
            lvl++;
        }
        return lvl;
    }

    // 插入节点
    public void insert(double score, String value) {
        SkipListNode[] update = new SkipListNode[MAX_LEVEL + 1];
        SkipListNode x = head;
        for (int i = level; i >= 0; i--) {
            while (x.forward[i] != null && x.forward[i].score < score) {
                x = x.forward[i];
            }
            update[i] = x;
        }
        int lvl = randomLevel();
        if (lvl > level) {
            for (int i = level + 1; i <= lvl; i++) {
                update[i] = head;
            }
            level = lvl;
        }
        x = new SkipListNode(lvl, score, value);
        for (int i = 0; i <= lvl; i++) {
            x.forward[i] = update[i].forward[i];
            update[i].forward[i] = x;
        }
    }

    // 删除节点
    public void delete(double score, String value) {
        SkipListNode[] update = new SkipListNode[MAX_LEVEL + 1];
        SkipListNode x = head;
        for (int i = level; i >= 0; i--) {
            while (x.forward[i] != null && x.forward[i].score < score) {
                x = x.forward[i];
            }
            update[i] = x;
        }
        x = x.forward[0];
        if (x != null && x.score == score && x.value.equals(value)) {
            for (int i = 0; i <= level; i++) {
                if (update[i].forward[i] != x) break;
                update[i].forward[i] = x.forward[i];
            }
            while (level > 0 && head.forward[level] == null) {
                level--;
            }
        }
    }

    // 查找节点
    public SkipListNode search(double score) {
        SkipListNode x = head;
        for (int i = level; i >= 0; i--) {
            while (x.forward[i] != null && x.forward[i].score < score) {
                x = x.forward[i];
            }
        }
        x = x.forward[0];
        if (x != null && x.score == score) {
            return x;
        }
        return null;
    }
}

示例使用

下面是一个简单的示例,展示如何使用上述跳表实现插入、删除和查找操作。

java 复制代码
public class SkipListExample {
    public static void main(String[] args) {
        SkipList skipList = new SkipList();

        // 插入节点
        skipList.insert(3.0, "Alice");
        skipList.insert(1.0, "Bob");
        skipList.insert(2.0, "Charlie");

        // 查找节点
        SkipListNode node = skipList.search(2.0);
        if (node != null) {
            System.out.println("Found node: " + node.value);
        } else {
            System.out.println("Node not found.");
        }

        // 删除节点
        skipList.delete(1.0, "Bob");

        // 查找被删除的节点
        node = skipList.search(1.0);
        if (node != null) {
            System.out.println("Found node: " + node.value);
        } else {
            System.out.println("Node not found.");
        }
    }
}

Redis 中使用跳表的优势

  1. 时间复杂度:跳表的插入、删除和查找操作的平均时间复杂度都是 (O(\log n)),与平衡树相当,但实现更为简单。
  2. 空间复杂度:跳表在最坏情况下的空间复杂度为 (O(n \log n)),但在实际使用中,空间效率较高。
  3. 实现简单:相对于红黑树和 AVL 树等平衡树,跳表的实现较为简单,易于理解和调试。

Redis 选择跳表作为有序集合的数据结构,主要是因为跳表在实现排序、范围查询等操作时,性能优越且实现简单。希望这段代码和解释能帮助你更好地理解跳表的原理和实现。

相关推荐
沉舟侧畔千帆过_15 小时前
一个DBA的真心话:搞定Oracle+PG双库,我就靠这招
数据库·oracle·dba
醉风塘15 小时前
【终极解决方案】Oracle ORA-01795错误:IN列表1000条限制的全面突破指南
数据库·oracle
信创天地15 小时前
从 Oracle 到国产数据库:迁移后成本直降 60%、性能反超 30% 的实战秘籍
数据库·oracle
Mikhail_G15 小时前
Mysql数据库操作指南——排序(零基础篇十)
大数据·数据库·sql·mysql·数据分析
沉舟侧畔千帆过_15 小时前
能源核心系统国产化攻坚:智能电网调度系统从 Oracle 到金仓 KES 迁移实录
数据库·oracle·能源·kingbase·金仓数据库
chengrise15 小时前
Oracle EBS 成本异常排查全指南:差异分摊、成本回滚场景与解决方案
运维·数据库·oracle·erp·ebs
wxc09015 小时前
Oracle 性能分析系列:tkprof 工具详解 —— 解码 10046 Trace 的利器
数据库·oracle
洁辉15 小时前
Oracle 数据库中将某个字段的值根据逗号(,)分隔符拆分成多行显示
数据库·oracle
qife12215 小时前
CVE-2026-21962漏洞利用工具:Oracle WebLogic代理插件未授权RCE检测与利用
数据库·网络安全·oracle·渗透测试·weblogic·cve-2026-21962
weixin_3906029915 小时前
产品实测:金仓与Oracle兼容性验证
数据库·oracle