日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

给你30s,如何跟面试官讲清楚跳表

發(fā)布時間:2024/1/18 编程问答 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 给你30s,如何跟面试官讲清楚跳表 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

查找

假設(shè)有如下這樣一個有序鏈表:

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

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

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

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

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

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

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

回顧下鏈表的定義:

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

我們在每一個節(jié)點的基礎(chǔ)上添加一個?down?指針,用來指向下一層的節(jié)點

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

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

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

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

具體的搜索算法如下:

/* 如果存在 x, 返回 x 所在的節(jié)點, 否則返回 x 的后繼節(jié)點 */ 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; }

插入

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

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

所以,我們在插入數(shù)據(jù)的時候,索引節(jié)點也需要相應(yīng)的改變來避免查找效率的退化

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

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

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

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

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

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

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

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

那插入數(shù)據(jù)時維護(hù)索引的時間復(fù)雜度是多少呢?

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

刪除

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

刪除元素的過程跟查找元素的過程類似,只不過在查找的路徑上如果發(fā)現(xiàn)了要刪除的元素 x,則執(zhí)行刪除操作。

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

總結(jié)

以上是生活随笔為你收集整理的给你30s,如何跟面试官讲清楚跳表的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。