Skip to content

Latest commit

 

History

History
142 lines (112 loc) · 4.53 KB

README.md

File metadata and controls

142 lines (112 loc) · 4.53 KB

设计跳表

难度:困难

https://leetcode.cn/problems/design-skiplist/

题目

不使用任何库函数,设计一个 跳表

跳表 是在 O(log(n)) 时间内完成增加、删除、搜索操作的数据结构。跳表相比于树堆与红黑树,其功能与性能相当,并且跳表的代码长度相较下更短,其设计思想与链表相似。

例如,一个跳表包含 [30, 40, 50, 60, 70, 90] ,然后增加 8045 到跳表中,以下图的方式操作:

image

Artyom Kalinin [CC BY-SA 3.0], via Wikimedia Commons

跳表中有很多层,每一层是一个短的链表。在第一层的作用下,增加、删除和搜索操作的时间复杂度不超过 O(n)。跳表的每一个操作的平均时间复杂度是 O(log(n)),空间复杂度是 O(n)

了解更多 : https://en.wikipedia.org/wiki/Skip_list

在本题中,你的设计应该要包含这些函数:

  • bool search(int target) : 返回target是否存在于跳表中。
  • void add(int num): 插入一个元素到跳表。
  • bool erase(int num): 在跳表中删除一个值,如果 num 不存在,直接返回false. 如果存在多个 num ,删除其中任意一个即可。

注意,跳表中可能存在多个相同的值,你的代码需要处理这种情况。

示例:

输入
["Skiplist", "add", "add", "add", "search", "add", "search", "erase", "erase", "search"]
[[], [1], [2], [3], [0], [4], [1], [0], [1], [1]]
输出
[null, null, null, null, false, null, true, false, true, false]

解释
Skiplist skiplist = new Skiplist();
skiplist.add(1);
skiplist.add(2);
skiplist.add(3);
skiplist.search(0);   // 返回 false
skiplist.add(4);
skiplist.search(1);   // 返回 true
skiplist.erase(0);    // 返回 false,0 不在跳表中
skiplist.erase(1);    // 返回 true
skiplist.search(1);   // 返回 false,1 已被擦除

解题

const MAX_LEVEL = 32
const P_FACTOR = 0.25

const randomLevel = () => {
  let level = 1
  // 随机生成
  while (Math.random() < P_FACTOR && level < MAX_LEVEL)
    level++

  return level
}

/**
 * 直接构造
 */
export class Skiplist {
  private head = new SkiplistNode(-1, MAX_LEVEL)
  private level = 0

  search(target: number): boolean {
    let curr: SkiplistNode | null = this.head
    for (let i = this.level - 1; i >= 0; i--) {
      // 找到第 i 层小于且最接近 target 的元素
      while (curr?.forward[i] && curr.forward[i]!.val < target)
        curr = curr.forward[i]
    }

    curr = curr?.forward[0] || null
    // 检测当前元素的值是否等于target
    return curr?.val === target
  }

  add(num: number): void {
    const update = new Array(MAX_LEVEL).fill(this.head)
    let curr: SkiplistNode | null = this.head
    for (let i = this.level - 1; i >= 0; i--) {
      // 找到第 i 层小于且最接近 num 的元素
      while (curr?.forward[i] && curr?.forward[i]!.val < num)
        curr = curr.forward[i]
      update[i] = curr
    }
    const level = randomLevel()
    this.level = Math.max(this.level, level)
    const newNode = new SkiplistNode(num, level)
    for (let i = 0; i < level; i++) {
      // 对第 i 层的状态进行更新,将当前元素的 forward 指向新的节点
      newNode.forward[i] = update[i].forward[i]
      update[i].forward[i] = newNode
    }
  }

  erase(num: number): boolean {
    const update = new Array<SkiplistNode | null>(MAX_LEVEL).fill(null)
    let curr: SkiplistNode | null = this.head
    for (let i = this.level - 1; i >= 0; i--) {
      // 找到第 i 层小于且最接近 num 的元素
      while (curr?.forward[i] && curr.forward[i]!.val < num)
        curr = curr.forward[i]
      update[i] = curr
    }

    curr = curr!.forward[0]
    // 如果值不存在则返回false
    if (!curr || curr.val !== num) return false

    for (let i = 0; i < this.level; i++) {
      if (update[i]?.forward[i] !== curr) break
      // 对第 i 层的状态进行更新,将 forward 指向被删除节点的下一跳
      update[i]!.forward[i] = curr.forward[i]
    }

    // 更新当前的level
    while (this.level > 1 && !this.head.forward[this.level - 1])
      this.level--

    return true
  }
}

class SkiplistNode {
  forward: (SkiplistNode | null)[]
  constructor(public val: number, maxLevel: number) {
    this.forward = new Array(maxLevel).fill(null)
  }
}