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

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

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

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

C語言編程學(xué)習(xí)基地 ? 來源:C語言編程學(xué)習(xí)基地 ? 作者:C語言編程學(xué)習(xí)基地 ? 2022-05-24 16:27 ? 次閱讀

雙向循環(huán)鏈表和它名字的表意一樣,就是把雙向鏈表的兩頭連接,使其成為了一個環(huán)狀鏈表。只需要將表中最后一個節(jié)點(diǎn)的next指針指向頭節(jié)點(diǎn),頭節(jié)點(diǎn)的prior指針指向尾節(jié)點(diǎn),鏈表就能成環(huán)兒,如圖所示:

c0a72b46-db39-11ec-ba43-dac502259ad0.jpg

需要注意的是,雖然雙向循環(huán)鏈表成環(huán)狀,但本質(zhì)上還是雙向鏈表,因此在雙向循環(huán)鏈表中,依然能夠找到頭指針和頭節(jié)點(diǎn)等。雙向循環(huán)鏈表和雙向鏈表相比,唯一的不同就是雙向循環(huán)鏈表首尾相連,其他都完全一樣。

注意:因為我上面已經(jīng)講了雙向鏈表,所以這里只注重講他們的實(shí)現(xiàn)差異。另因為帶頭節(jié)點(diǎn)會更好操作,所以我的代碼都有頭節(jié)點(diǎn)。

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

初始化時需要將頭節(jié)點(diǎn)的next和prior都指向自己。

c0be579e-db39-11ec-ba43-dac502259ad0.jpg

//1、初始化雙向循環(huán)鏈表(帶頭節(jié)點(diǎn))

Status initLinkList(LinkList *list){

//創(chuàng)建頭節(jié)點(diǎn)

*list = malloc(sizeof(Node));

if (*list == NULL) {

return ERROR;

}

//前驅(qū)和后繼都指向自己

(*list)->prior = *list;

(*list)->data = -1;

(*list)->next = *list;

printf("已初始化鏈表~ ");

return OK;

}

2、遍歷雙向循環(huán)鏈表

注意它的尾節(jié)點(diǎn)的next不再是Null,而是頭節(jié)點(diǎn)

//2、遍歷雙向循環(huán)鏈表

void printfLinkLisk(LinkList list){

printf("遍歷鏈表: ");

if (list == NULL || list->next == list) {

printf("這是一個空鏈表 ");

return;

}

LinkList p = list;

//判斷next是否全部正確

printf("根據(jù)next從前往后遍歷:");

while (p->next != list) {

printf("%d ",p->next->data);

p = p->next;

}

printf(" ");

//判斷prior是否全部正確

printf("根據(jù)prior從后往前遍歷:");

while (p != list) {

printf("%d ",p->data);

p = p->prior;

}

printf(" ");

}

3、根據(jù)索引位置添加節(jié)點(diǎn)

這里不需要判斷尾節(jié)點(diǎn)的next是否為Null,因為它會指向頭節(jié)點(diǎn)。

//3、根據(jù)索引位置插入數(shù)據(jù)至鏈表中

Status insertLinkList(LinkList *list, int index, ElemType data){

if (list == NULL || index < 0) {

return ERROR;

}

int i = 0;

LinkList priorNode = *list;

//判斷插入的位置,這里開始位置是0,index超過鏈表長度則插入末尾

while (i < index && priorNode->next != *list) {

priorNode = priorNode->next;

i++;

}

LinkList newNode = malloc(sizeof(Node));

if (newNode == NULL) {

return ERROR;

}

newNode->data = data;

//插入操作共四步,看好了,別眨眼

//1.將priorNode->next節(jié)點(diǎn)的前驅(qū)指向新節(jié)點(diǎn)

priorNode->next->prior = newNode;

//2.將新節(jié)點(diǎn)->next指向原來的priorNode->next

newNode->next = priorNode->next;

//3.將priorNode->next指向新節(jié)點(diǎn)

priorNode->next = newNode;

//4.新節(jié)點(diǎn)的前驅(qū)指向priorNode

newNode->prior = priorNode;

return OK;

}

4、根據(jù)索引位置刪除節(jié)點(diǎn)

這里不需要判斷尾節(jié)點(diǎn)的next是否為Null,因為它會指向頭節(jié)點(diǎn)。

//4、根據(jù)索引位置刪除節(jié)點(diǎn)

Status deleteLinkListByIndex(LinkList *list, int index, ElemType *data){

if (*list == NULL || index < 0) {

return ERROR;

}

LinkList locaNode = *list;

int i = 0;

//注意別刪了頭節(jié)點(diǎn)

while (i <= index) {

locaNode = locaNode->next;

if (locaNode == *list) {

printf("沒有這個你想要刪除的節(jié)點(diǎn) ");

return ERROR;

}

i++;

}

//開始刪除,只需要做兩步

locaNode->prior->next = locaNode->next;

locaNode->next->prior = locaNode->prior;

*data = locaNode->data;

free(locaNode);

return OK;

}

5、根據(jù)存儲的值刪除節(jié)點(diǎn)

這里不需要判斷尾節(jié)點(diǎn)的next是否為Null,因為它會指向頭節(jié)點(diǎn)。

//5、根據(jù)存儲的值刪除節(jié)點(diǎn)

Status deleteLinkListByData(LinkList *list, ElemType data){

if (*list == NULL) {

return ERROR;

}

LinkList locaNode = (*list)->next;

while (locaNode != *list) {

if (locaNode->data == data) {

break;

}

locaNode = locaNode->next;

}

if (locaNode == *list) {

printf("沒有這個你想要刪除的節(jié)點(diǎn) ");

return ERROR;

}

//開始刪除,只需要做兩步

locaNode->prior->next = locaNode->next;

locaNode->next->prior = locaNode->prior;

free(locaNode);

return OK;

}

6、根據(jù)值查找節(jié)點(diǎn)

尾節(jié)點(diǎn)的next可是頭節(jié)點(diǎn)哦,找到它就是最后一個了。

//6、查找元素

Status selectNode(LinkList list, ElemType data, LinkList *locaNode){

if (list == NULL) {

return ERROR;

}

LinkList p = list->next;

while (p != list) {

if (p->data == data) {

*locaNode = p;

break;

}

p = p->next;

}

if (*locaNode == NULL) {

printf("沒有這個你想要的節(jié)點(diǎn) ");

return ERROR;

}

else {

return OK;

}

}

其它代碼

#include "stdlib.h"

#define OK 1

#define ERROR 0

//元素類型

typedef int ElemType;

//狀態(tài)類型

typedef int Status;

//定義節(jié)點(diǎn)結(jié)構(gòu)體

typedef struct Node {

struct Node *prior;

ElemType data;

struct Node *next;

} Node;

typedef Node *LinkList;

int main(int argc, const char * argv[]) {

LinkList list;

initLinkList(&list);

for (int i = 0; i < 10; i ++) {

insertLinkList(&list, i, i);

}

printfLinkLisk(list);

int index, data;

printf("輸入你想插入的位置(從0開始)和存儲的值:");

scanf("%d %d",&index,&data);

insertLinkList(&list, index, data);

printfLinkLisk(list);

printf("輸入你想刪除的位置(從0開始):");

scanf("%d",&index);

deleteLinkListByIndex(&list, index, &data);

printfLinkLisk(list);

printf("輸入你想刪除的節(jié)點(diǎn)的值(只刪最前的那個):");

scanf("%d",&data);

deleteLinkListByData(&list, data);

printfLinkLisk(list);

printf(" ");

return 0;

}

輸出結(jié)果:

c0cf2b46-db39-11ec-ba43-dac502259ad0.jpg

—END—

審核編輯 :李倩

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

    關(guān)注

    0

    文章

    212

    瀏覽量

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

    關(guān)注

    0

    文章

    80

    瀏覽量

    10520

原文標(biāo)題:【C語言教程】“雙向循環(huán)鏈表”學(xué)習(xí)總結(jié)及其代碼實(shí)現(xiàn)!

文章出處:【微信號:cyuyanxuexi,微信公眾號:C語言編程學(xué)習(xí)基地】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    數(shù)組和鏈表在內(nèi)存中的區(qū)別 數(shù)組和鏈表的優(yōu)缺點(diǎn)

    數(shù)組和鏈表在內(nèi)存中的區(qū)別 數(shù)組和鏈表的優(yōu)缺點(diǎn)? 數(shù)組和鏈表是常見的數(shù)據(jù)結(jié)構(gòu),用于組織和存儲數(shù)據(jù)。它們在內(nèi)存中的存儲方式以及優(yōu)缺點(diǎn)方面存在一些顯著的差異。本文將詳細(xì)探討這些差異以及它們的優(yōu)缺點(diǎn)。 1.
    的頭像 發(fā)表于 02-21 11:30 ?651次閱讀

    數(shù)組和鏈表有何區(qū)別

    數(shù)組和鏈表的區(qū)別,這個問題,不僅面試中經(jīng)常遇到,考研的同學(xué)也得掌握才行。
    的頭像 發(fā)表于 02-19 15:33 ?364次閱讀
    數(shù)組和<b class='flag-5'>鏈表</b>有何區(qū)別

    循環(huán)指令loop規(guī)定循環(huán)次數(shù)

    循環(huán)指令是計算機(jī)編程中非常重要的概念,它允許程序重復(fù)執(zhí)行一段代碼塊,使得程序可以更有效地處理大量數(shù)據(jù)和重復(fù)性任務(wù)。在本文中,我們將詳盡、詳實(shí)、細(xì)致地介紹循環(huán)指令的相關(guān)概念、語法和應(yīng)用場
    的頭像 發(fā)表于 02-14 16:10 ?1025次閱讀

    數(shù)據(jù)結(jié)構(gòu):刪除有序鏈表的重復(fù)節(jié)點(diǎn)

    給定一個有序單鏈表(從小到大有序)的頭結(jié)點(diǎn)head(該結(jié)點(diǎn)有值),刪除鏈表中的重復(fù)元素,使鏈表中的所有元素都只出現(xiàn)一次。如當(dāng)輸入 {1,1,2} 時,經(jīng)刪除后,原鏈表變?yōu)?{1,2},
    的頭像 發(fā)表于 12-05 15:46 ?552次閱讀
    數(shù)據(jù)結(jié)構(gòu):刪除有序<b class='flag-5'>鏈表</b>的重復(fù)節(jié)點(diǎn)

    數(shù)據(jù)結(jié)構(gòu):判斷鏈表回文結(jié)構(gòu)

    給定一個鏈表,判斷該鏈表是否為回文結(jié)構(gòu)?;匚氖侵冈撟址蚰嫘蛲耆恢隆H绠?dāng)輸入鏈表 {1,2,3,2,1} 時,斷定是回文結(jié)構(gòu),輸出True。
    的頭像 發(fā)表于 12-01 13:26 ?476次閱讀
    數(shù)據(jù)結(jié)構(gòu):判斷<b class='flag-5'>鏈表</b>回文結(jié)構(gòu)

    數(shù)據(jù)結(jié)構(gòu):單鏈表的排序

    給定一個單鏈表的頭結(jié)點(diǎn)head(該結(jié)點(diǎn)有值),長度為n的無序單鏈表,對其按升序排序后,返回新鏈表。如當(dāng)輸入鏈表 {3,1,4,5,2} 時,經(jīng)升序排列后,原
    的頭像 發(fā)表于 11-30 13:56 ?1143次閱讀
    數(shù)據(jù)結(jié)構(gòu):單<b class='flag-5'>鏈表</b>的排序

    python循環(huán)創(chuàng)建變量并賦值

    循環(huán)是Python編程中非常重要的一個概念,它可以讓我們輕松地重復(fù)執(zhí)行某些代碼塊,從而簡化編程過程并提高代碼的效率。在循環(huán)中,我們經(jīng)常需要創(chuàng)建變量并賦值,這是非常常見的操作。接下來,我將詳盡地解釋在
    的頭像 發(fā)表于 11-23 14:51 ?1362次閱讀

    python怎么把for循環(huán)的值拿出來

    Python中可以使用for循環(huán)來遍歷一個序列或者迭代器中的元素。當(dāng)我們希望將for循環(huán)中的值取出來并進(jìn)行其他操作時,我們可以使用一些方法和技巧來實(shí)現(xiàn)。 一、使用列表解析 列表解析是一種創(chuàng)建新列表
    的頭像 發(fā)表于 11-22 09:54 ?2545次閱讀

    for in range循環(huán)怎么使用

    for-in range 循環(huán)是Python中的一種循環(huán)結(jié)構(gòu),用于重復(fù)執(zhí)行一段代碼,而且循環(huán)次數(shù)是已知的。 在Python中,for-in range 循環(huán)有以下幾種用法: 通過指定
    的頭像 發(fā)表于 11-21 14:49 ?1.1w次閱讀

    請問鏈表是怎么用的?

    鏈表是怎么用的?好像單片機(jī)很少用到這種數(shù)據(jù)結(jié)構(gòu),平時應(yīng)用在在哪里比較多
    發(fā)表于 11-08 06:41

    C語言中鏈表的作用是什么?

    對C語言中指針用的很少,鏈表、文件操作幾乎沒用過,所以也不能理解到底有什么作用。各位有經(jīng)常在做程序時會用到這些嗎。
    發(fā)表于 11-06 06:23

    請問鏈表在單片機(jī)C語言中有應(yīng)用嗎?

    鏈表在單片機(jī)C語言中有應(yīng)用么?
    發(fā)表于 10-16 07:28

    LinkedBlockingQueue基于單向鏈表的實(shí)現(xiàn)

    的 LinkedBlockingQueue。它的底層基于單向鏈表實(shí)現(xiàn)。 先看一看它的 Node 內(nèi)部類和主要屬性、構(gòu)造函數(shù)。 Node static class Node E > { E item; Node next; Node
    的頭像 發(fā)表于 10-13 11:41 ?541次閱讀
    LinkedBlockingQueue基于單向<b class='flag-5'>鏈表</b>的實(shí)現(xiàn)

    鏈表在單片機(jī)上為什么用的不多?

    鏈表在單片機(jī)上為什么用的不多
    發(fā)表于 10-07 08:03

    鏈表在單片機(jī)上效率高嗎?

    鏈表在單片機(jī)上效率高么?
    發(fā)表于 09-26 08:01