阿里一面:跳表的原理和时间复杂度,为什么还需要字典结构配合?

文章内容收录到个人网站,方便阅读http://hardyfish.top/

跳表(SkipList)是 一种有序链表的多级索引结构,通过增加多层索引来加快查找速度。

  • 最底层:完整的有序链表(包含所有元素)。
  • 上层索引:抽取部分节点作为快捷通道,减少查找步数。
  • 查找过程:从最高层索引开始,能向右就向右,不能向右就向下,直到找到目标或最接近的位置。

举例(每层越上节点越少):

Level 3:  1 ---------  9
Level 2:  1 --- 5 ---  9
Level 1:  1  3  5  7  9

找元素 7 的过程:

  • 从 Level 3 的 1 → 9(超了,往下)
  • Level 2 的 1 → 5 → 9(超了,往下)
  • Level 1 的 5 → 7(找到)

时间复杂度

跳表的平均性能和 平衡二叉树(AVL/红黑树) 类似:

查找

O(log N)

O(N)

插入/删除

O(log N)

O(N)

范围查询

O(log N + m)

O(N)

m 为返回的结果数量。

在随机化层数 + 合理概率(Redis 用 1/4)下,O(N) 出现概率极低。

为什么 ZSet 还需要 字典结构(hash) 配合

在 Redis 的有序集合 ZSet 里,底层是 跳表(skiplist) + 字典(dict) 组合:

  • 跳表: 按分值(score)排序。支持范围查询(如 ZRANGEBYSCORE)。插入、删除、按排名查询效率高。
  • 字典: key:成员(member)value:分值(score)O(1) 时间 根据成员查分值(比如 ZSCORE、ZREM 先定位成员)。跳表按 score 排序,没法快速通过成员查 score,所以需要字典辅助。

为什么不只用跳表?

  • 如果只用跳表,查一个成员的分值需要 O(logN),而有了字典就是 O(1)。
  • 删除成员时,需要先 O(1) 定位到 score,再 O(logN) 从跳表删除,整体性能更好。

总结一句话

跳表保证了 有序存储 + 范围查询的高效性,字典保证了 按成员快速定位

Redis 的 ZSet 用 双结构冗余存储,是空间换时间的设计,既能 O(1) 找成员,又能 O(logN) 做有序操作。

#面试题##面试#
大厂面试每日一题 文章被收录于专栏

大厂每日一道面试题!

全部评论

相关推荐

09-07 15:20
门头沟学院 Java
点赞 评论 收藏
分享
投递淘天集团等公司10个岗位
点赞 评论 收藏
分享
评论
1
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务