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

【數據結構之鏈表】詳細圖文教你花樣玩鏈表

開發 前端
單鏈表的每個結點只有一個數據域和一個指針域,而該指針域只存儲了下一個結點的地址,所以我們只能通過某結點找到其直接后繼結點,卻不能通過某節點找到其直接前驅結點。

[[392598]]

0. 提要鉤玄

 前面在文章【數據結構之鏈表】看完這篇文章我終于搞懂鏈表了中已經介紹了鏈式存儲結構,介紹了鏈式存儲結構的最基本(簡單)實現——單向鏈表。

單向鏈表,顧名思義,它是單向的。

因為單鏈表的每個結點只有一個數據域和一個指針域,而該指針域只存儲了下一個結點的地址,所以我們只能通過某結點找到其直接后繼結點,卻不能通過某節點找到其直接前驅結點。

此外,由于單鏈表到尾結點(鏈表的最后一個結點)結束,所以尾結點的指針域是 NULL,以此來表示鏈表的終止,這就導致我們遍歷到尾結點的時候,如果想再次遍歷,只能手動回到頭結點再開始遍歷。

為了彌補單鏈表的上面兩個缺點,下面介紹兩種鏈表,它們都是單鏈表的變形,如果你理解了單鏈表,那么會很容易理解這兩種變形。

 

目錄

1. 單向循環鏈表

1.1. 結構

單鏈表的尾結點的指針域是 NULL,所以單鏈表到此終止。如果我們使用單鏈表的尾結點的指針域存儲頭結點的地址,即尾結點的直接后繼結點為頭結點,如此一來,單鏈表就構成了一個環(循環),稱之為單項循環鏈表。


1.2. 實現思路

單向循環鏈表是由單鏈表進化而來的,算是單鏈表的“兒子”,所以單鏈表的那一套結構對于單向循環鏈表來說完全適用,從上圖你也可以看出,結構并無較大改變,二者所不同只在尾結點,所以我們只需要在尾結點和與尾結點相關的操作上下功夫就行了。

因此,單向循環鏈表的結構體和單鏈表的結構體相同。

  1. /*單向循環鏈表的結點的結構體*/ 
  2. typedef struct _Node { 
  3.     int data; //數據域:存儲數據 
  4.     struct _Node *next; //指針域:存儲直接后繼結點的地址 
  5. } Node; 

為了統一對空鏈表和非空鏈表的操作,我們使用帶頭結點的鏈表來實現它。

1.3. 空鏈表及初始化

一個空鏈表如圖所示,只有一個頭指針和頭結點:

空鏈表

頭結點的指針域指向其本身構成一個循環,我們可以借此來判斷鏈表是否為空。

  1. if (head->next == head) { 
  2.     printf("空鏈表。\n"); 

想要初始化一個空鏈表很簡單,創造頭結點,使頭結點的 next 指針指向其自身即可:

  1. Node *create_node(int elem) 
  2.     Node *new = (Node *) malloc(sizeof(Node)); 
  3.     new->data = elem; 
  4.     new->next = NULL
  5.     return new; 
  6.  
  7. /** 
  8.  * 初始化鏈表 
  9.  * p_head: 指向頭指針的指針 
  10.  */ 
  11. void init(Node **p_head) 
  12.     //創建頭結點 
  13.     Node *head_node = create_node(0); 
  14.     //頭指針指向頭結點 
  15.     *p_head = head_node; 
  16.     //頭結點的next指針指向其本身,構成環 
  17.     head_node->next = head_node; 

1.4. 插入操作

這里只演示頭插法和尾插法

【頭插法】

因為帶頭結點,所以不需要考慮是否為空鏈表。下圖是向空鏈表中頭插兩個元素的過程:

單向循環鏈表頭插法過程

  1. /** 
  2.  * 頭插法,新結點為頭結點的直接后繼 
  3.  * p_head: 指向頭指針的指針 
  4.  * elem: 新結點的數據 
  5.  */ 
  6. void insert_at_head(Node **p_head, int elem) 
  7.     Node *new = create_node(elem); 
  8.     Node *head_node = *p_head; //頭結點 
  9.     //新結點插入頭結點之后 
  10.     new->next = head_node->next
  11.     head_node->next = new; 

【尾插法】

因為為了盡量簡單,所以我們并沒有設置指向尾結點的尾指針,所以在尾插之前,需要先借助某個指針,遍歷至尾結點,然后再插入。

  1. /** 
  2.  * 尾插法:新插入的結點始終在鏈表尾 
  3.  * p_head: 指向頭指針的指針 
  4.  * elem: 新結點的數據 
  5.  */ 
  6. void insert_at_tail(Node **p_head, int elem) 
  7.     Node *new = create_node(elem); 
  8.     Node *head_node = *p_head; //頭結點 
  9.     Node *tail = head_node; //tail指針指向頭結點 
  10.     while (tail->next != head_node) { //tail遍歷至鏈表尾 
  11.         tail = tail->next
  12.     } 
  13.     //尾插 
  14.     new->next = tail->next
  15.     tail->next = new; 

1.5. 刪除操作

刪除的本質是“跳過”待刪除的結點,所以我們要找到待刪除結點的直接前驅結點,然后讓其直接前驅結點的 next 指針指向其直接后繼結點,以此來“跳過”待刪除結點,最后保存其數據域,釋放結點,即完成刪除。

這里只演示頭刪法。

因為刪除的是頭結點的直接后繼結點,所以我們不必再費力尋找待刪除結點的直接前驅結點了。

單向循環鏈表頭刪法過程

  1. /** 
  2.  * 頭刪法:刪除頭結點之后的結點 
  3.  * p_head: 指向頭指針的指針 
  4.  * elem: 指向保存數據變量的指針 
  5.  */ 
  6. void delete_from_head(Node **p_head, int *elem) 
  7.     Node *head_node = *p_head; //頭結點 
  8.     if (head_node->next == head_node) { 
  9.         printf("空鏈表,無元素可刪。\n"); 
  10.         return
  11.     } 
  12.     Node *first_node = head_node->next; //首結點:頭結點的下一個結點 
  13.     *elem = first_node->data; //保存被刪除結點的數據 
  14.     head_node->next = first_node->next; //刪除結點 
  15.     free(first_node); //釋放 

1.6. 遍歷操作

我們可以一圈又一圈地循環遍歷鏈表,下面是循環打印 20 次結點地代碼:

  1. /** 
  2.  * 循環打印20次結點 
  3.  */ 
  4. void output_20(Node *head) 
  5.     if (head->next == head) { 
  6.         printf("空鏈表。\n"); 
  7.         return
  8.     } 
  9.     Node *p = head->next
  10.     for (int i = 0; i <= 20; i++) { 
  11.         if (p != head) { //不打印頭結點 
  12.             printf("%d ", p->data); 
  13.         } 
  14.         p = p->next
  15.     } 
  16.     printf("\n"); 

2. 雙向鏈表

2.1. 結構

顧名思義,雙向鏈表,就是有兩個方向,一個指向前,一個指向后。這樣我們就彌補了單鏈表的某個結點只能找到其直接后繼的缺陷。如圖所示:

雙向鏈表

2.2. 實現思路

為了實現能指前和指后的效果,只靠 next 指針肯定是不夠的,所以我們需要再添加一個指針 —— prev,該指針指向某結點的直接前驅結點。

  1. /*雙向鏈表的結點結構體*/ 
  2. typedef struct _Node { 
  3.     int data; //數據域 
  4.     struct _Node *prev; //指向直接前驅結點的指針 
  5.     struct _Node *next; //指向直接后繼結點的指針 
  6. } Node; 

2.3. 空鏈表及初始化

雙向鏈表的空鏈表如圖所示:

 

雙向空鏈表

要初始化一個這樣的空鏈表,需要創造出頭結點,然后將兩個指針域置空即可:

  1. Node *create_node(int elem) 
  2.     Node *new = (Node *)malloc(sizeof(Node)); 
  3.     new->data = elem; 
  4.     new->prev = NULL
  5.     new->next = NULL
  6.     return new; 
  7.  
  8. void init(Node **p_head) 
  9.     //創建頭結點 
  10.     Node *head_node = create_node(0); 
  11.     //頭指針指向頭結點 
  12.     *p_head = head_node; 

2.4. 插入操作

這里只演示頭插法,過程如下:

雙向鏈表頭插法過程

代碼如下:

  1. /** 
  2.  * 頭插法,新結點為頭結點的直接后繼 
  3.  * p_head: 指向頭指針的指針 
  4.  * elem: 新結點的數據 
  5.  */ 
  6. void insert_at_head(Node **p_head, int elem) 
  7.     Node *new = create_node(elem); 
  8.     Node *head_node = *p_head; //頭結點 
  9.     if (head_node->next != NULL) { //不為空鏈表 
  10.         Node *first_node = head_node->next; //首結點:頭結點的下一個結點 
  11.         //首結點的prev指針指向new結點 
  12.         first_node->prev = new; 
  13.         //new結點的next指針指向首結點 
  14.         new->next = first_node; 
  15.     } 
  16.     //new結點的prev指針指向頭結點 
  17.     new->prev = head_node; 
  18.     //頭結點的next指針指向new結點 
  19.     head_node->next = new; 

2.5. 刪除操作

這里只演示頭刪法。下圖是將一個有兩個元素結點的雙向鏈表頭刪為空鏈表的過程:

雙向鏈表頭刪法過程

代碼如下:

  1. /** 
  2.  * 頭刪法 
  3.  * p_head: 指向頭指針的指針 
  4.  * elem: 指向保存變量的指針 
  5.  */ 
  6. void delete_from_head(Node **p_head, int *elem) 
  7.     Node *head_node = *p_head; //頭結點 
  8.     Node *first_node = head_node->next; //待刪除的首結點:頭結點的下一個結點 
  9.     if (head_node->next == NULL) { //判空 
  10.         printf("空鏈表,無元素可刪。\n"); 
  11.         return
  12.     } 
  13.     *elem = first_node->data; //保存數據 
  14.      
  15.     if (first_node->next != NULL) { 
  16.         first_node->next->prev = first_node->prev; 
  17.     } 
  18.     first_node->prev->next = first_node->next
  19.     free(first_node); 

2.6. 遍歷操作

有了 next 指針域,我們可以一路向后遍歷;有了 prev 指針域,我們可以一路向前遍歷。

這里不再展示代碼了。

3. 總結

了解了單向循環鏈表和雙向鏈表,就像拿搭積木一樣,我們還可以創造出來雙向循環鏈表。這里就不再演示了,讀者可以自行嘗試。只要你搞懂上面三種鏈表,這絕非難事。

以上就是鏈表的花樣玩法部分內容,以后還會繼續更新。

參考資料

[1]GitHub: https://github.com/xingrenguanxue/Simple-DS-and-Easy-Algo

[2]Gitee: https://gitee.com/xingrenguanxue/Simple-DS-and-Easy-Algo

 

責任編輯:姜華 來源: 二十二畫程序員
相關推薦

2021-07-13 07:52:03

Python數據結構

2021-07-15 06:43:12

Python數據結構

2017-03-01 13:58:46

Python數據結構鏈表

2021-08-03 10:24:59

數據跳躍鏈表結構

2012-02-02 10:21:05

單鏈表nexthead

2021-05-12 14:09:35

鏈表數據結構線性結構

2021-12-21 08:19:29

數據結構算法鏈表相交

2021-10-29 11:27:52

鏈表數據結構算法

2021-01-06 08:03:00

JavaScript數據結構

2021-01-28 07:33:34

JavaScript鏈表數據

2021-03-10 08:42:19

Java數據結構算法

2018-08-01 15:22:36

數據科學家數據科學分析

2016-11-10 08:44:45

數據挖掘分析

2021-03-11 08:53:20

Java數據結構算法

2020-10-28 10:10:03

Java單鏈表數據結構

2023-10-06 20:21:28

Python鏈表

2020-03-27 14:29:30

數據結構

2021-04-30 15:06:34

鴻蒙HarmonyOS應用

2021-05-10 15:05:56

鴻蒙HarmonyOS應用

2021-01-21 08:23:29

鏈表單鏈表循環鏈表
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久99国产精一区二区三区 | 亚洲精品天堂 | 国产精品一区二区久久 | 亚洲国产小视频 | 看片网站在线 | 久久手机视频 | 精品亚洲永久免费精品 | 国产精品观看 | 一本综合久久 | 国产免费看 | 成人免费视频一区二区 | 一级全黄少妇性色生活免费看 | 日韩在线欧美 | 国产精彩视频 | 亚洲综合色视频在线观看 | 在线观看日韩 | 一区二区精品视频 | 成人精品在线视频 | 日韩午夜 | 久久久亚洲综合 | 97在线观视频免费观看 | 国产激情三区 | 欧美视频二区 | 日本一区二区三区四区 | 国产亚洲精品美女久久久久久久久久 | 九九热在线视频 | 偷拍自拍网 | 欧美在线a| 涩涩视频在线看 | 91九色婷婷 | 欧美日韩综合 | 成人在线免费观看视频 | 亚洲444eee在线观看 | 久久亚洲一区二区三区四区 | 日韩电影免费在线观看中文字幕 | 午夜男人视频 | 成人不卡 | 国产高清在线精品一区二区三区 | 久久精品一 | 亚洲一区二区免费 | 午夜免费网站 |