上節(jié)說過這節(jié)會講雙向鏈表,環(huán)形鏈表和應(yīng)用舉例,我們開始吧!?。。?/p> 首先,明白什么是雙向鏈表。所謂雙向鏈表是如果希望找直接前驅(qū)結(jié)點(diǎn)和直接后繼結(jié)點(diǎn)的時(shí)間復(fù)雜度都是 O(1),那么,需要在結(jié)點(diǎn)中設(shè)兩個(gè)引用域,一個(gè)保存直接前驅(qū)結(jié)點(diǎn)的地址,叫 prev,一個(gè)直接后繼結(jié)點(diǎn)的地址,叫 next,這樣的鏈表就是雙向鏈表(Doubly Linked List)。雙向鏈表的結(jié)點(diǎn)結(jié)構(gòu)示意圖如圖所示。 雙向鏈表結(jié)點(diǎn)的定義與單鏈表的結(jié)點(diǎn)的定義很相似, ,只是雙向鏈表多了一個(gè)字段 prev。其實(shí),雙向鏈表更像是一根鏈條一樣,你連我,我連你,不清楚,請看圖。 雙向鏈表結(jié)點(diǎn)類的實(shí)現(xiàn)如下所示 //一個(gè)鏈條的類 public class DbNode<T> //當(dāng)前的數(shù)據(jù)所在 說了這么多雙向鏈表接點(diǎn)的類的屬性,我們要看一看他的相關(guān)的操作。這里只做一些畫龍點(diǎn)睛地方的描述 插入操作:設(shè) p是指向雙向鏈表中的某一結(jié)點(diǎn),即 p存儲的是該結(jié)點(diǎn)的地址,現(xiàn)要將一個(gè)結(jié)點(diǎn) s 插入到結(jié)點(diǎn) p 的后面,插入的源代碼如下所示:操作如下: p.Next.Prev = s; 插入過程如圖所示(以 p 的直接后繼結(jié)點(diǎn)存在為例) 。 注意:引用域值的操作的順序不是唯一的,但也不是任意的,操作?必須放到操作?的前面完成,否則 p 直接后繼結(jié)點(diǎn)的就找不到了。這一點(diǎn)需要讀者把每個(gè)操作的含義搞清楚。此算法時(shí)間操作消耗在查找上,其時(shí)間的復(fù)雜度是O(n). 下面,看他的刪除操作,以在結(jié)點(diǎn)之后刪除為例來說明在雙向鏈表中刪除結(jié)點(diǎn)的情況。 設(shè) p是指向雙向鏈表中的某一結(jié)點(diǎn),即 p存儲的是該結(jié)點(diǎn)的地址,現(xiàn)要將一個(gè)結(jié)點(diǎn) s插入到結(jié)點(diǎn) p的后面 。偽代碼如下:操作如下: p.Next = P.Next.Next; 刪除過程如圖所示(以 p的直接后繼結(jié)點(diǎn)存在為例) 相應(yīng)的算法的時(shí)間復(fù)雜度也是消耗到結(jié)點(diǎn)的查找上,其復(fù)雜度應(yīng)該是O(n) 查找操作與單鏈表的極其的類似,也是從頭開始遍歷。相應(yīng)偽代碼如圖所示: current.next=p.next.next current.prev=p.next.prev; 相應(yīng)的偽代碼如下圖所示: 該算法的時(shí)間復(fù)雜度,是一個(gè)個(gè)的遍歷的過程中,顧時(shí)間復(fù)雜度是O(n) 獲取當(dāng)前的雙向鏈表長度與 查找類似,不做過多的贅述,這里,我們把雙向鏈表基本概念和操作基本介紹完了,下面介紹一個(gè)重要的鏈表——環(huán)形鏈表。 首先,還是老樣子,看看環(huán)形鏈表的定義。有些應(yīng)用不需要鏈表中有明顯的頭尾結(jié)點(diǎn)。在這種情況下,可能需要方便地從最后一個(gè)結(jié)點(diǎn)訪問到第一個(gè)結(jié)點(diǎn)。此時(shí),最后一個(gè)結(jié)點(diǎn)的引用域不是空引用,而是保存的第一個(gè)結(jié)點(diǎn)的地址(如果該鏈表帶結(jié)點(diǎn),則保存的是頭結(jié)點(diǎn)的地址) ,也就是頭引用的值。我們把這樣的鏈表結(jié)構(gòu)稱之為環(huán)形鏈表。他就像小朋友手拉手做游戲。如圖所示。 用鏈表如圖所示: 這里基本添加,刪除,操作的操作與單鏈表簡直是一模一樣,這里就沒有必要寫這些東西。我們主要看他們一些簡單應(yīng)用。 應(yīng)用舉例一 已知單鏈表 H,寫一算法將其倒置,即實(shí)現(xiàn)如圖所示的操作,其中(a)為倒置前,(b)為倒置后。 算法思路:由于單鏈表的存儲空間不是連續(xù)的,所以,它的倒置不能像順表那樣,把第 i 個(gè)結(jié)點(diǎn)與第 n-i 個(gè)結(jié)點(diǎn)交換(i 的取值范圍是 1 到 n/2,n 為單鏈表的長度) 。其解決辦法是依次取單鏈表中的每個(gè)結(jié)點(diǎn)插入到新鏈表中去。并且,為了節(jié)省內(nèi)存資源,把原鏈表的頭結(jié)點(diǎn)作為新鏈表的頭結(jié)點(diǎn)。存儲整數(shù)的單鏈表的倒置的算法實(shí)現(xiàn)如下: public void ReversLinkList(LinkList<int> H) 舉例2,約瑟夫環(huán)問題,題目如下: 已知n個(gè)人(以編號1,2,3...n分別表示)圍坐在一張圓桌周圍。從編號為k的人開始報(bào)數(shù),數(shù)到m的那個(gè)人出列;他的下一個(gè)人又從1開始報(bào)數(shù),數(shù)到m的那個(gè)人又出列;依此規(guī)律重復(fù)下去,直到圓桌周圍的人全部出列。求最后出列的人相應(yīng)的編號。 void JOSEPHUS(int n,int k,int m) //n為總?cè)藬?shù),k為第一個(gè)開始報(bào)數(shù)的人,m為出列者喊到的數(shù) { /* p為當(dāng)前結(jié)點(diǎn) r為輔助結(jié)點(diǎn),指向p的前驅(qū)結(jié)點(diǎn) list為頭節(jié)點(diǎn)*/ LinkList p,r,list; /*建立循環(huán)鏈表*/ for(int i=0;i<n;i++) { p=(LinkList)LNode; p.data=i; if(list==NULL) list=p; else r.link=p; r=p; } p.link=list; /*使鏈表循環(huán)起來*/ p=list; /*使p指向頭節(jié)點(diǎn)*/ /*把當(dāng)前指針移動到第一個(gè)報(bào)數(shù)的人*/ for(i=0;i<k;i++) { r=p; p=p.link; } /*循環(huán)地刪除隊(duì)列結(jié)點(diǎn)*/ while(p.link!=p) { for(i=0;i<m-1;i++) { r=p; p=p.link; } r.link=p.link; console.writeline("被刪除的元素:{0} ",p.data); free(p); p=r.node.; } console.writeLine("\n最后被刪除的元素是:{0}",P.data); 具體的算法,如圖所示:
這個(gè)算法的時(shí)間的復(fù)雜度是O(n2)
} 還和大家分享的一個(gè)例子,就是我做做一個(gè)類似與網(wǎng)易郵箱的產(chǎn)品時(shí)候,幾千萬甚至數(shù)以億級的大數(shù)量登錄的時(shí)候,發(fā)現(xiàn)用戶登錄的時(shí)候真他媽的慢,你猜我開始是怎么做的,就是直接查數(shù)據(jù)庫,這當(dāng)然是不行的。這怎么辦了, 最后,我在一個(gè)高人的指教下,發(fā)現(xiàn)登錄的時(shí)候速度飛快,怎么搞的。我把所有的數(shù)據(jù)庫的數(shù)據(jù)讀入到內(nèi)存中,然后把數(shù)據(jù)用鏈表把他們串起來,到我查詢某個(gè)用戶時(shí)候,只比較用戶的 字節(jié)數(shù)。 這就是我眼中的鏈表結(jié)構(gòu)。
|
|