0

0

JS如何实现跳表?跳表的插入和删除

畫卷琴夢

畫卷琴夢

发布时间:2025-08-25 13:37:01

|

613人浏览过

|

来源于php中文网

原创

跳表通过多层级链表和随机化层级设计,在平均情况下实现O(logN)的查找、插入和删除性能,其核心优势在于实现简单、并发性能好、缓存友好,且适用于有序数据的高效操作,常见于Redis有序集合等场景。

js如何实现跳表?跳表的插入和删除

跳表(Skip List)在JavaScript中实现,本质上是构建一个多层级的链表结构。它的核心思想是通过概率性地在不同层级维护有序的链表,从而在平均情况下实现对数时间复杂度的查找、插入和删除操作,性能上可以媲美平衡二叉搜索树,但在实现上却简单得多。它的插入和删除操作都依赖于先找到元素的位置,然后像操作普通链表一样调整指针,只不过这个过程需要在多个层级上同步进行。

解决方案

实现跳表,我们首先需要一个节点(Node)结构,它包含值、以及一个指向多层下一个节点的数组(

next
)。然后是跳表类本身,它管理着头节点、最大层级以及一个随机层级生成器。

class SkipListNode {
    constructor(value, level) {
        this.value = value;
        // next是一个数组,存储指向不同层级下一个节点的引用
        this.next = new Array(level + 1).fill(null);
    }
}

class SkipList {
    constructor(maxLevel = 16, probability = 0.5) {
        this.maxLevel = maxLevel; // 跳表的最大层级
        this.probability = probability; // 决定节点层级的概率因子
        this.level = 0; // 当前跳表的最高层级
        // 头节点,其值通常为null或-Infinity,用于简化边界处理
        this.head = new SkipListNode(null, maxLevel); 
    }

    // 随机生成新节点的层级
    // 这是一个核心机制,决定了跳表的性能
    randomLevel() {
        let lvl = 0;
        while (Math.random() < this.probability && lvl < this.maxLevel) {
            lvl++;
        }
        return lvl;
    }

    // 插入操作
    insert(value) {
        // update数组用来存储每一层需要更新的节点
        // update[i] 表示在第i层,新节点应该插入到update[i]的后面
        const update = new Array(this.maxLevel + 1).fill(null);
        let current = this.head;

        // 从最高层开始向下查找,找到插入位置
        for (let i = this.level; i >= 0; i--) {
            while (current.next[i] && current.next[i].value < value) {
                current = current.next[i];
            }
            update[i] = current; // 记录下当前层的前一个节点
        }

        // 如果值已经存在,通常选择不插入或更新,这里选择不插入
        if (current.next[0] && current.next[0].value === value) {
            // console.log(`Value ${value} already exists.`);
            return false;
        }

        // 确定新节点的层级
        const newLevel = this.randomLevel();

        // 如果新节点的层级高于当前跳表的最高层级,需要更新head的next指针
        // 并且update数组中超出当前level的部分,其前驱节点就是head
        if (newLevel > this.level) {
            for (let i = this.level + 1; i <= newLevel; i++) {
                update[i] = this.head;
            }
            this.level = newLevel; // 更新跳表的最高层级
        }

        // 创建新节点
        const newNode = new SkipListNode(value, newLevel);

        // 调整指针,将新节点插入到相应的位置
        for (let i = 0; i <= newLevel; i++) {
            newNode.next[i] = update[i].next[i];
            update[i].next[i] = newNode;
        }
        return true;
    }

    // 删除操作
    delete(value) {
        const update = new Array(this.maxLevel + 1).fill(null);
        let current = this.head;

        // 从最高层开始向下查找,找到要删除的节点
        for (let i = this.level; i >= 0; i--) {
            while (current.next[i] && current.next[i].value < value) {
                current = current.next[i];
            }
            update[i] = current; // 记录下当前层的前一个节点
        }

        // 检查要删除的节点是否存在
        current = current.next[0];
        if (!current || current.value !== value) {
            // console.log(`Value ${value} not found.`);
            return false;
        }

        // 调整指针,跳过要删除的节点
        for (let i = 0; i <= this.level; i++) {
            // 如果update[i]的下一个节点是要删除的节点,就跳过它
            if (update[i].next[i] === current) {
                update[i].next[i] = current.next[i];
            }
        }

        // 删除后,检查是否需要降低跳表的最高层级
        // 从最高层开始检查,如果head的next指针指向null,说明该层已空
        while (this.level > 0 && this.head.next[this.level] === null) {
            this.level--;
        }
        return true;
    }

    // 查找操作(通常也会实现,但这里不作为重点)
    search(value) {
        let current = this.head;
        for (let i = this.level; i >= 0; i--) {
            while (current.next[i] && current.next[i].value < value) {
                current = current.next[i];
            }
        }
        current = current.next[0];
        return current && current.value === value;
    }

    // 打印跳表(辅助调试)
    print() {
        console.log("Skip List:");
        for (let i = this.level; i >= 0; i--) {
            let current = this.head.next[i];
            let levelStr = `Level ${i}: Head -> `;
            while (current) {
                levelStr += `${current.value} -> `;
                current = current.next[i];
            }
            levelStr += "NULL";
            console.log(levelStr);
        }
    }
}

// 示例用法:
// const skipList = new SkipList();
// skipList.insert(3);
// skipList.insert(6);
// skipList.insert(7);
// skipList.insert(9);
// skipList.insert(12);
// skipList.insert(1);
// skipList.print();
// console.log("Searching for 7:", skipList.search(7)); // true
// console.log("Searching for 5:", skipList.search(5)); // false
// skipList.delete(7);
// skipList.print();
// console.log("Searching for 7 after deletion:", skipList.search(7)); // false
// skipList.delete(1);
// skipList.print();
// skipList.delete(100); // Value 100 not found.

跳表为什么能比平衡树更快?它的核心优势在哪里?

对我来说,跳表最吸引人的地方,是它在实现复杂度和性能之间的那种微妙平衡。我们都知道,平衡二叉搜索树(比如红黑树、AVL树)在理论上提供了严格的O(logN)性能保证,但它们的实现,尤其是插入和删除后的“旋转”和“着色”操作,那真的是相当烧脑,调试起来更是痛苦。相较之下,跳表的核心优势就在于它的概率性结构和实现上的简洁性

首先,实现难度大大降低。跳表不需要复杂的平衡算法。插入时,你只需要通过一个简单的随机函数来决定新节点的层级,然后像操作链表一样插入;删除时也类似,找到节点后直接调整指针即可。这种“简单粗暴”的方式,在工程实践中意味着更少的bug、更快的开发周期。我个人就觉得,与其花大量时间去搞懂红黑树的各种旋转规则,不如用跳表,效率上差不太多,但省心太多了。

其次,并发性能上的潜在优势。在多线程或并发环境下,跳表在某些操作上表现得比平衡树更好。因为它的结构是多层链表,在进行插入或删除时,往往只需要锁定少量相关的节点,而不是像平衡树那样可能需要对整个子树进行复杂的全局性调整。这种局部性锁定的特性,使得跳表在并发数据结构的设计中非常受欢迎,比如Redis的Sorted Set就是基于跳表实现的。

此外,缓存友好性也是一个不容忽视的优点。跳表的节点在内存中通常是连续的,或者至少比二叉树的节点分布更线性。这有助于CPU缓存的命中率,因为处理器在访问数据时,往往会预取相邻的数据。虽然这不总是绝对的优势,但在某些场景下,它确实能带来实际的性能提升。平衡树的节点可能散落在内存的各个角落,导致更多的缓存未命中。

最后,虽然是概率性的,但跳表在平均情况下的性能是非常可靠的O(logN)。只要随机函数足够好,你几乎可以总是获得与平衡树相媲美的性能。这种“足够好”的随机性,对于大多数应用场景来说已经足够了。

在实际项目中,跳表有哪些常见的应用场景?

跳表虽然不如哈希表或平衡树那么“家喻户晓”,但在一些特定领域,它可是实实在在的“幕后英雄”。它简洁高效的特性,让它在需要有序数据且对插入/删除性能有较高要求的场景下,显得格外有用。

最典型的应用,莫过于数据库索引。比如,大名鼎鼎的Redis,它的有序集合(Sorted Set)就是通过跳表来实现的。有序集合需要支持快速地按分数范围查询、添加、删除元素,并且能按序遍历。跳表完美契合了这些需求:查找、插入、删除都是对数时间复杂度,同时还能高效地进行范围查询(因为数据在每一层都是有序的)。这比使用哈希表(无法保持顺序)或单纯的链表(查找慢)要高效得多。

除了数据库,并发数据结构也是跳表大展拳脚的地方。正如前面提到的,跳表的局部性锁定优势,使得它非常适合构建无锁(lock-free)或读写锁(read-write lock)优化的并发数据结构。在高性能计算、高并发服务中,如果需要一个有序的集合,并且要处理大量的并发读写请求,跳表会是一个非常好的选择。它能够减少线程间的竞争,提高系统的吞吐量。

司马诸葛
司马诸葛

基于企业知识文档,就可训练专属AI数字员工

下载

再往深一点看,一些网络路由表的实现也可能借鉴跳表的思想。路由表需要快速查找IP地址对应的下一跳,并且路由规则可能会动态添加或删除。跳表的多层级结构和高效的查找能力,使其在处理这种有序查找和更新的场景时具有优势。

甚至在一些内存管理垃圾回收算法中,如果需要维护一个有序的空闲内存块列表,跳表也可以用来高效地管理这些内存块,以便快速分配和回收。

总结来说,只要你的项目需要一个能够快速查找、插入、删除,并且数据需要保持有序的数据结构,同时你又希望实现起来相对简单,或者对并发性能有较高要求,那么跳表就非常值得考虑。它不像那些“万金油”的数据结构,但它在自己的“舒适区”里,表现是相当出色的。

实现跳表时,有哪些常见的“坑”或者需要特别注意的技术细节?

实现跳表,虽然整体上比平衡树简单,但它也有一些自己的“脾气”和需要注意的细节,不然一不小心就会踩坑。我自己在写的时候,就遇到过一些小问题,值得拿出来聊聊。

首先,随机层级生成器的质量至关重要。跳表的性能在很大程度上依赖于这个随机性。如果你的随机函数不够“随机”,或者概率因子设置不合理,可能会导致跳表退化成普通链表(所有节点都在第一层),或者层级过高(浪费内存)。通常我们用

Math.random() < probability
来决定是否增加层级,这个
probability
(通常是0.5)需要根据实际情况和经验来设置。如果这个概率太低,节点层级普遍不高,跳表会比较“扁”,查找性能可能受影响;如果太高,节点层级普遍很高,虽然查找快,但内存开销会变大。

其次,

update
数组的正确使用是插入和删除操作的关键。这个数组在查找过程中,记录了每一层需要更新的“前驱节点”。在插入时,新节点要插入到
update[i]
update[i].next[i]
之间;在删除时,
update[i].next[i]
需要跳过被删除的节点,直接指向被删除节点的下一个节点。如果这里处理不当,比如数组索引越界,或者指针链断裂,整个跳表就可能崩溃。尤其是在新节点的层级高于当前跳表最高层级时,
update
数组中超出原
level
的部分,其前驱节点都应该是
head
,这个细节很容易被忽略。

还有一个小点,就是头节点(

head
)的处理。我通常会给头节点一个
null
-Infinity
的值,并且它的层级设置为
maxLevel
。这样做的好处是,头节点总是在所有元素的“前面”,并且它的
next
数组总是有足够的空间来容纳指向最高层级节点的指针。这样可以避免在处理边界情况时写出很多额外的判断逻辑,代码会显得更简洁。

最后,删除操作后最高层级的维护。当你删除一个节点后,如果这个节点恰好是某个层级的唯一节点,或者它被删除后导致最高层级变得空荡荡(

head.next[this.level]
变成了
null
),那么你需要适时地降低跳表的
this.level
。这虽然不是性能上的大问题,但可以避免跳表维持过高的空层级,节省一点内存,也让结构看起来更“紧凑”。不处理这个,跳表也能正常工作,但从“工程美学”上讲,稍微有些不完美。

这些细节,看似微不足道,但在实际编码中,它们往往是导致bug或者让代码变得晦涩难懂的罪魁祸首。理解并正确处理它们,才能真正发挥跳表的优势。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

536

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

372

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

706

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

470

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

989

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

652

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

537

2023.09.20

笔记本电脑卡反应很慢处理方法汇总
笔记本电脑卡反应很慢处理方法汇总

本专题整合了笔记本电脑卡反应慢解决方法,阅读专题下面的文章了解更多详细内容。

1

2025.12.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号