成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

關于跳表,這么解釋你肯定能聽懂

數據庫 其他數據庫
跳表中,每一層索引都是一個有序的單鏈表,單鏈表刪除元素的時間復雜度為 O(1),最多需要刪除 LogN 個元素(索引層數為 LogN),所以刪除元素的總時間包 = 查找元素的時間 + 刪除 LogN 個元素的時間 = O(LogN ) + O(LogN ) = 2O(LogN ),忽略常數部分,刪除元素的時間復雜度為 O(LogN)。

查找

假設有如下這樣一個有序鏈表:

圖片

想要查找 24、43、59,按照順序遍歷,分別需要比較的次數為 2、4、6

目前查找的時間復雜度是 O(N),如何提高查找效率?

很容易想到二分查找,將查找的時間復雜度降到 O(LogN)

具體來說,我們把鏈表中的一些節點提取出來,作為索引,類似于二叉搜索樹,得到如下結構:

圖片

這里我們把 10、30、50、80 提取出來作為一級索引,這樣搜索的時候就可以使用二分查找來減少比較次數了。

我們還可以再從一級索引提取一些元素出來,作為二級索引,變成如下結構:

圖片

比如如果想要查找 59,那么搜索路徑就是下面這樣的:

圖片

回顧下鏈表的定義:

class ListNode {
private int val;
private ListNode next;

public ListNode(int val){
this.val = val;
this.next = null;
}
}

我們在每一個節點的基礎上添加一個 down 指針,用來指向下一層的節點

class Node {
private int val;
private ListNode next;
private ListNode down;

public ListNode(int val){
this.val = val;
this.next = null;
this.down = null;
}
}

這樣,一個最簡單的跳表節點就定義出來了。

我們這里說的只是最簡單的實現,像比如 Redis 的跳表實現和我們說的還是有所不同的,當然了,思想都是一致的

所以跳表是什么?簡單來說,跳表就是支持二分查找的有序鏈表

具體的搜索算法如下:

/* 如果存在 x, 返回 x 所在的節點, 否則返回 x 的后繼節點 */  
private Node find(x){
p = top;
while (true) {
while (p.next.val < x){
p = p.next;
}
if (p.down == null){
return p.next;
}
p = p.down;
}
return null;
}

插入

關于插入,大家可能很容易想到往最下面一層的有序鏈表中添加數據,但是索引該咋辦?索引要不要更新呢?

如果不更新索引,就可能出現兩個索引節點之間數據非常多的情況,極端情況下跳表就會退化為單鏈表,從而使得查找效率從 O(LogN) 退化為 O(N)。

所以,我們在插入數據的時候,索引節點也需要相應的改變來避免查找效率的退化

比較容易想到的做法就是完全重建索引,我們每次插入數據后,都把這個跳表的索引刪掉全部重建。因為索引的空間復雜度是 O(N),即:索引節點的個數是 O(N) 級別,每次完全重新建一個 O(N) 級別的索引,時間復雜度也是 O(N) 。造成的后果是:為了維護索引,導致每次插入數據的時間復雜度變成了 O(N)。

那有沒有其他效率比較高的方式來維護索引呢?

最理想的索引就是在原始鏈表中每隔一個元素抽取一個元素做為一級索引。換種說法,我們在原始鏈表中【隨機】的選 n/2 個元素做為一級索引是不是也能通過索引提高查找的效率呢?

當然可以,因為一般隨機選的元素相對來說都是比較均勻的。如下圖所示,隨機選擇了 n/2 個元素做為一級索引,雖然不是每隔一個元素抽取一個,但是對于查找效率來講,影響不大,比如我們想找元素 16,仍然可以通過一級索引,使得遍歷路徑較少了將近一半。

圖片

當然了,如果抽取的一級索引的元素恰好是前一半的元素 1、3、4、5、7、8,那么查找效率確實沒有提升,但是這樣的概率太小了。所以我們可以認為:當原始鏈表中元素數量足夠大,且抽取足夠隨機的話,我們得到的索引是均勻的。所以,我們可以維護一個這樣的索引:隨機選 n/2? 個元素做為一級索引、隨機選 n/4? 個元素做為二級索引、隨機選 n/8 個元素做為三級索引,依次類推,一直到最頂層索引。這里每層索引的元素個數已經確定,且每層索引元素選取的足夠隨機,所以可以通過索引來提升跳表的查找效率。

那代碼具體該如何實現,使得在每次新插入元素的時候,盡量讓該元素有 1/2 的幾率建立一級索引、1/4 的幾率建立二級索引、1/8 的幾率建立三級索引....呢?

其實很簡單啦,搞一個概率算法就行了(具體是怎么個概率法這里就不詳細解釋了),當每次有數據要插入時,先通過概率算法告訴我們這個元素需要插入到幾級索引中,然后開始維護索引并把數據插入到原始鏈表中。

如下所示,插入新元素 12,假設概率算法返回的結果是 4,表示新元素需要插入到 4 級索引中,同時,我們還需要建立 3 級索引、2 級索引和 1 級索引(也就是原始有序鏈表)

圖片

那插入數據時維護索引的時間復雜度是多少呢?

跳表中,每一層索引都是一個有序的單鏈表,元素插入到單鏈表的時間復雜度為 O(1),我們索引的高度最多為 LogN,當插入一個元素 x 時,最壞的情況就是元素 x 需要插入到每層索引中,所以插入數據的最壞時間復雜度是 O(LogN),最好的時間復雜度是 O(1)。

刪除

跳表刪除數據時,要把索引中對應節點也要刪掉。如下圖所示,如果要刪除元素 8,需要把原始鏈表中的 8 和第 2、3 級索引的 8 都刪除掉。

圖片

刪除元素的過程跟查找元素的過程類似,只不過在查找的路徑上如果發現了要刪除的元素 x,則執行刪除操作。

跳表中,每一層索引都是一個有序的單鏈表,單鏈表刪除元素的時間復雜度為 O(1),最多需要刪除 LogN 個元素(索引層數為 LogN),所以刪除元素的總時間包 = 查找元素的時間 + 刪除 LogN 個元素的時間 = O(LogN ) + O(LogN ) = 2O(LogN ),忽略常數部分,刪除元素的時間復雜度為 O(LogN)。

責任編輯:武曉燕 來源: 飛天小牛肉
相關推薦

2021-03-19 06:08:09

智慧城市物聯網城市服務

2016-08-04 16:30:49

華為

2019-07-01 05:02:34

IP地址子網掩碼 網關

2020-09-23 10:24:45

數據庫SQL技術

2022-05-18 18:31:28

機器人自然語言編程

2011-06-14 11:15:13

Qt 面試題 函數指針

2020-11-12 10:01:55

人工智能

2023-11-10 08:22:09

雪花算法生成算法分布式

2023-06-02 13:27:27

ChatGPT模型

2020-08-17 09:35:03

安卓手機流暢iPhone

2023-05-14 23:39:51

機器人深度學習

2024-04-19 14:27:26

檢索增強生成大型語言模型

2025-05-12 08:55:00

2021-09-05 23:54:55

人工智能機器語言

2019-06-26 09:41:44

分布式事務微服務

2020-11-02 12:47:56

性能優化

2024-04-08 11:13:27

AIEVI人工智能

2023-06-07 07:43:06

APIVue 2Vue 3

2010-07-17 01:12:31

Telnet服務

2025-04-09 10:16:29

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 一区二区三区在线 | 欧 | 国产精品久久性 | 在线视频91| 在线黄 | 久久精品超碰 | 色婷婷av一区二区三区软件 | 狠狠爱一区二区三区 | 伊人无码高清 | 亚洲一区二区三区桃乃木香奈 | a视频在线观看 | 国产精品久久久久久婷婷天堂 | 国产精品久久久久久久久久不蜜臀 | 亚洲欧美精品国产一级在线 | 中文字幕亚洲无线 | 国产精品一区二区三区在线 | 国产一区二区电影网 | 99精品99| 亚洲综合无码一区二区 | 91av视频在线观看 | 亚洲精品成人 | 日本男人天堂 | 综合久久久 | 成人精品一区二区 | 国产高清精品在线 | 久久乐国产精品 | 亚洲国产一区二区在线 | 亚洲视频精品 | 日本免费一区二区三区四区 | 精品一区二区三区四区 | 91麻豆精品国产91久久久更新资源速度超快 | 麻豆久久 | 国产三级网站 | 久久久久国产精品 | 男人av在线播放 | 国精品一区 | 日操操| 视频一区二区在线观看 | 国产精品日韩在线观看 | 久久se精品一区精品二区 | 天堂中文在线观看 | 一区欧美 |