0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

周立功新著內(nèi)容分享:雙向鏈表是什么?

電子工程師 ? 來(lái)源:互聯(lián)網(wǎng) ? 作者:佚名 ? 2017-09-22 18:24 ? 次閱讀

近日周立功教授公開了數(shù)年的心血之作《程序設(shè)計(jì)與數(shù)據(jù)結(jié)構(gòu)》,電子版已無(wú)償性分享到電子工程師與高校群體下載,經(jīng)周立功教授授權(quán),特對(duì)本書內(nèi)容進(jìn)行連載。

>>>>1.1 雙向鏈表

單向鏈表的添加、刪除操作,都必須找到當(dāng)前結(jié)點(diǎn)的上一個(gè)結(jié)點(diǎn),以便修改上一個(gè)結(jié)點(diǎn)的p_next指針完成相應(yīng)的操作。由于單向鏈表的結(jié)點(diǎn)沒有指向其上一個(gè)結(jié)點(diǎn)的指針,因此只有從頭結(jié)點(diǎn)開始遍歷鏈表。當(dāng)某一結(jié)點(diǎn)的p_next指向當(dāng)前結(jié)點(diǎn)時(shí),表明它為當(dāng)前結(jié)點(diǎn)的上一個(gè)結(jié)點(diǎn)。顯然每次都要從頭開始遍歷,其效率極為低下。在單向鏈表中,之所以可以直接獲取單向鏈表中當(dāng)前結(jié)點(diǎn)的下一個(gè)結(jié)點(diǎn),是因?yàn)榻Y(jié)點(diǎn)中包含了指向下一個(gè)結(jié)點(diǎn)的指針p_next。如果在雙向鏈表的結(jié)點(diǎn)中再增加一個(gè)指向它的前一個(gè)結(jié)點(diǎn)的前向指針p_prev,則一切問題將迎刃而解。那么,既有指向下一個(gè)結(jié)點(diǎn)的指針,又有指向前一個(gè)結(jié)點(diǎn)的指針的鏈表稱之為雙向鏈表,示意圖詳見圖3.15。

圖3.15 雙向鏈表示意圖

與單向鏈表一樣,雙向鏈表也定義了一個(gè)頭結(jié)點(diǎn),基于單向鏈表將應(yīng)用數(shù)據(jù)與鏈表結(jié)構(gòu)相關(guān)數(shù)據(jù)完全分離的設(shè)計(jì)思想,則雙向鏈表結(jié)點(diǎn)僅保留p_next和p_prev指針。其數(shù)據(jù)結(jié)構(gòu)定義如下:

typedef struct _dlist_node{

struct _dlist_node *p_next;

struct _dlist_node *p_prev;

}dlist_node_t;

其中,dlist是double list 的縮寫,表明該結(jié)點(diǎn)是雙向鏈表結(jié)點(diǎn)。由此可見,雖然前向指針使得尋找鏈表的上一個(gè)結(jié)點(diǎn)變得非常容易,但由于結(jié)點(diǎn)中新增了一個(gè)指針,因此其內(nèi)存開銷將會(huì)是單向鏈表的兩倍。在實(shí)際應(yīng)用中,應(yīng)該權(quán)衡效率與內(nèi)存空間,在內(nèi)存資源非常緊缺的場(chǎng)合,如果結(jié)點(diǎn)的添加、刪除操作很少,一點(diǎn)效率的影響可以接受,則選擇使用單向鏈表。而不是一味地追求效率,認(rèn)為雙向鏈表比單向鏈表好,始終選擇使用雙向鏈表。

圖3.15中,頭結(jié)點(diǎn)的p_prev和尾結(jié)點(diǎn)的p_next直接被設(shè)置為了NULL,此時(shí),如果要直接由頭結(jié)點(diǎn)找到尾結(jié)點(diǎn),或者由尾結(jié)點(diǎn)找到頭結(jié)點(diǎn),都必須遍歷整個(gè)鏈表。可以對(duì)這兩個(gè)指針稍加利用,使頭結(jié)點(diǎn)的p_prev指向尾結(jié)點(diǎn),尾結(jié)點(diǎn)的p_next指向頭結(jié)點(diǎn),此時(shí),該雙向鏈表就成了一個(gè)循環(huán)雙向鏈表,示意圖詳見圖3.16。

圖3.16 循環(huán)雙向鏈表示意圖

由于循環(huán)雙向鏈表的效率更高,可以直接從頭結(jié)點(diǎn)找到尾結(jié)點(diǎn),或者從尾結(jié)點(diǎn)找到頭結(jié)點(diǎn),且沒有額外的內(nèi)存空間消耗,僅僅是使用了兩個(gè)不打算使用的指針,算是廢物利用,因此下面介紹的雙向鏈表均視為循環(huán)雙向鏈表。

類似于單向鏈表,雖然頭結(jié)點(diǎn)與普通結(jié)點(diǎn)的內(nèi)容完全相同,但它們的含義卻有所區(qū)別,頭結(jié)點(diǎn)是鏈表的頭,代表了整個(gè)鏈表,擁有此頭結(jié)點(diǎn),就表示其擁有了整個(gè)鏈表。為了便于區(qū)分頭結(jié)點(diǎn)與普通結(jié)點(diǎn),可以單獨(dú)定義一個(gè)頭結(jié)點(diǎn)類型。比如:

typedef dlist_node_t dlist_head_t;

當(dāng)需要使用雙向鏈表時(shí),首先需要使用該類型定義一個(gè)頭結(jié)點(diǎn)。比如:

dlist_head_t head;

由于此時(shí)還沒有添加其它任何結(jié)點(diǎn),僅存在一個(gè)頭結(jié)點(diǎn),因此該頭結(jié)點(diǎn)既是第一個(gè)結(jié)點(diǎn)(頭結(jié)點(diǎn)),又是最后一個(gè)結(jié)點(diǎn)(尾結(jié)點(diǎn))。按照循環(huán)鏈表的定義,尾結(jié)點(diǎn)的p_next指向頭結(jié)點(diǎn),頭結(jié)點(diǎn)的p_prev指向尾結(jié)點(diǎn),僅有一個(gè)結(jié)點(diǎn)的示意圖詳見圖3.17。

圖3.17 空鏈

顯然,僅有頭結(jié)點(diǎn)時(shí),其p_next和p_prev都指向本身。即:

head.p_next = &head;

head.p_prev = &head;

為了避免用戶直接操作成員,需要定義一個(gè)初始化函數(shù),專門用于初始化鏈表頭結(jié)點(diǎn)中各個(gè)成員的值,其函數(shù)原型(dlist.h)為:

int dlist_init(dlist_head_t *p_head);

其中,p_head指向待初始化的鏈表頭結(jié)點(diǎn)。其調(diào)用形式如下:

dlist_head_t head;

dlist_init(&head);

dlist_init()函數(shù)的實(shí)現(xiàn)詳見程序清單3.33。

程序清單3.33雙向鏈表初始化函數(shù)

1 int dlist_init(dlist_head_t *p_head)

2 {

3 if (p_head == NULL){

4 return -1;

5 }

6 p_head -> p_next = p_head;

7 p_head -> p_prev = p_head;

8 return 0;

9 }

與單向鏈表類似,將提供一些基礎(chǔ)的操作接口,它們的函數(shù)原型如下:

dlist_node_t *dlist_prev_get (dlist_head_t *p_head, dlist_node_t *p_pos); //尋找某一結(jié)點(diǎn)的前一結(jié)點(diǎn)

dlist_node_t *dlist_next_get (dlist_head_t *p_head, dlist_node_t *p_pos); //尋找某一結(jié)點(diǎn)的后一結(jié)點(diǎn)

dlist_node_t *dlist_tail_get (dlist_head_t *p_head); //獲取尾結(jié)點(diǎn)

dlist_node_t *dlist_begin_get (dlist_head_t *p_head); //獲取開始位置,第一個(gè)用戶結(jié)點(diǎn)

dlist_node_t *dlist_end_get (dlist_head_t *p_head); //獲取結(jié)束位置,尾結(jié)點(diǎn)下一個(gè)結(jié)點(diǎn)的位置

對(duì)于dlist_prev_get()和dlist_next_get(),在鏈表結(jié)點(diǎn)中已經(jīng)存在指向前驅(qū)后后繼的指針,詳見程序清單3.34。

程序清單3.34得到結(jié)點(diǎn)前驅(qū)和后繼的函數(shù)實(shí)現(xiàn)

1 dlist_node_t *dlist_prev_get(dlist_head_t *p_head, dlist_node_t *p_pos)

2 {

3 if (p_pos != NULL){

4 return p_pos -> p_prev;

5 }

6 return NULL;

7 }

8

9 dlist_node_t *dlist_next_get(dlist_head_t *p_head, dlist_node_t *p_pos)

10 {

11 if (p_pos != NULL){

12 return p_pos -> p_next;

13 }

14 return NULL;

15 }

dlist_tail_get()函數(shù)用于得到鏈表的尾結(jié)點(diǎn),在循環(huán)雙向鏈表中,頭結(jié)點(diǎn)的p_reev即指向了尾結(jié)點(diǎn)詳見程序清單3.35。

程序清單3.35 dlist_tail_get()函數(shù)實(shí)現(xiàn)

1 dlist_node_t *dlist_tail_get(dlist_head_t *p_head)

2 {

3 if (p_head != NULL) {

4 return p_head->p_prev;

5 }

6 return NULL;

7 }

dlist_begin_get()函數(shù)用于得到第一個(gè)用戶結(jié)點(diǎn),詳見程序清單3.36。

程序清單3.36 dlist_begin_get()函數(shù)實(shí)現(xiàn)

1 dlist_node_t *dlist_begin_get(dlist_head_t *p_head)

2 {

3 if (p_head != NULL){

4 return p_head->p_next;

5 }

6 return NULL;

7 }

dlist_end_get()用于得到鏈表的結(jié)束位置,當(dāng)雙向鏈表設(shè)計(jì)為循環(huán)雙向鏈表時(shí),則頭結(jié)點(diǎn)的p_prev和尾結(jié)點(diǎn)的p_next都被有效地利用了,任何有效結(jié)點(diǎn)的p_nextp_prev都不再為NULL。顯然,不能再以NULL作為結(jié)束位置了,當(dāng)從第一個(gè)結(jié)點(diǎn)開始順序訪問鏈表的各個(gè)結(jié)點(diǎn)時(shí),尾結(jié)點(diǎn)的下一個(gè)結(jié)點(diǎn)就是鏈表頭結(jié)點(diǎn)(head),因此結(jié)束位置就是頭結(jié)點(diǎn)本身。dlist_end_get()的實(shí)現(xiàn)詳見程序清單3.37。

程序清單3.37 dlist_end_get()函數(shù)實(shí)現(xiàn)

1 dlist_node_t *dlist_end_get(dlist_head_t *p_head)

2 {

3 if (p_head != NULL) {

4 return p_head->p_prev;

5 }

6 return NULL;

7 }

在公眾號(hào)后臺(tái)回復(fù)關(guān)鍵字“程序設(shè)計(jì)”,即可在線閱讀《程序設(shè)計(jì)與數(shù)據(jù)結(jié)構(gòu)》;回復(fù)關(guān)鍵字“編程”,即可在線閱讀《面向AMetal框架與接口的編程(上)》。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • mcu
    mcu
    +關(guān)注

    關(guān)注

    146

    文章

    17171

    瀏覽量

    351483
  • 周立功
    +關(guān)注

    關(guān)注

    38

    文章

    130

    瀏覽量

    37660
  • 鏈表
    +關(guān)注

    關(guān)注

    0

    文章

    80

    瀏覽量

    10572

原文標(biāo)題:周立功:雙向鏈表是什么?

文章出處:【微信號(hào):Zlgmcu7890,微信公眾號(hào):周立功單片機(jī)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    立功闡釋高效的雙向鏈表如何用

    實(shí)際上循環(huán)鏈表,無(wú)論是頭結(jié)點(diǎn)、尾結(jié)點(diǎn)還是普通結(jié)點(diǎn),其本質(zhì)上都是一樣的。
    的頭像 發(fā)表于 09-25 14:14 ?6500次閱讀

    立功教授談迭代器模式設(shè)計(jì)

    近日立功教授公開了數(shù)年的心血之作《程序設(shè)計(jì)與數(shù)據(jù)結(jié)構(gòu)》,電子版已無(wú)償性分享到電子工程師與高校群體下載,經(jīng)立功教授授權(quán),特對(duì)本書內(nèi)容進(jìn)行連
    的頭像 發(fā)表于 09-26 13:51 ?6306次閱讀
    <b class='flag-5'>周</b><b class='flag-5'>立功</b>教授談迭代器模式設(shè)計(jì)

    立功大師EASY FPGA原理圖

    本帖最后由 eehome 于 2013-1-5 09:47 編輯 立功EASYFPGA原理圖立功大師經(jīng)典力作,F(xiàn)PGA原理圖。歡迎大家下載學(xué)習(xí)
    發(fā)表于 03-16 11:02

    立功的NIOS視頻

    立功的NIOS視頻
    發(fā)表于 07-19 09:55

    立功CANTest軟件

    立功CANTest軟件
    發(fā)表于 01-15 16:52

    立功CANTest軟件

    立功CANTest軟件
    發(fā)表于 02-27 09:26

    【HarmonyOS】雙向循環(huán)鏈表

    了一個(gè)個(gè)雙向循環(huán)鏈表,把指針的高效能運(yùn)用到了極致,這也許就是編程的藝術(shù)吧!致敬鴻蒙內(nèi)核開發(fā)者貢獻(xiàn)了如此優(yōu)秀的源碼,鴻蒙內(nèi)核源碼可作為大學(xué)C語(yǔ)言,操作系統(tǒng),數(shù)據(jù)結(jié)構(gòu)三門課的教學(xué)項(xiàng)目
    發(fā)表于 10-20 15:39

    立功Labvie demo分離ID和CAN內(nèi)容

    請(qǐng)教,立功 Labview demo中怎么將CAN 的ID和內(nèi)容分別讀取出來(lái)。
    發(fā)表于 04-04 17:16

    立功ARM培訓(xùn)精華

    立功ARM培訓(xùn)精華 ppt模式,還需下載分段壓縮才可以查看
    發(fā)表于 02-11 09:13 ?108次下載

    TinyM0配套教程大全 立功

    TinyM0配套教程大全 立功
    發(fā)表于 04-20 16:30 ?0次下載

    立功ARM培訓(xùn)精華

    立功ARM培訓(xùn)精華,有需要的下來(lái)看看。
    發(fā)表于 01-13 17:23 ?41次下載

    算法與數(shù)據(jù)結(jié)構(gòu)——雙向鏈表

    第三章為算法與數(shù)據(jù)結(jié)構(gòu),本文為3.3 雙向鏈表。
    的頭像 發(fā)表于 09-19 17:56 ?7301次閱讀
    算法與數(shù)據(jù)結(jié)構(gòu)——<b class='flag-5'>雙向</b><b class='flag-5'>鏈表</b>

    了解Linux通用的雙向循環(huán)鏈表

    在linux內(nèi)核中,有一種通用的雙向循環(huán)鏈表,構(gòu)成了各種隊(duì)列的基礎(chǔ)。鏈表的結(jié)構(gòu)定義和相關(guān)函數(shù)均在include/linux/list.h中,下面就來(lái)全面的介紹這一鏈表的各種API。
    發(fā)表于 05-07 10:44 ?678次閱讀

    單片機(jī)教程 立功

    單片機(jī)教程 立功
    發(fā)表于 11-19 14:17 ?0次下載

    雙向循環(huán)鏈表的創(chuàng)建

    需要注意的是,雖然雙向循環(huán)鏈表成環(huán)狀,但本質(zhì)上還是雙向鏈表,因此在雙向循環(huán)鏈表中,依然能夠找到頭
    的頭像 發(fā)表于 05-24 16:27 ?2117次閱讀