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

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

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

C語言-鏈表(單向鏈表、雙向鏈表)

DS小龍哥-嵌入式技術(shù) ? 來源:DS小龍哥-嵌入式技術(shù) ? 作者:DS小龍哥-嵌入式技 ? 2022-09-09 11:30 ? 次閱讀

1. 鏈表結(jié)構(gòu)介紹

在前面章節(jié)已經(jīng)學(xué)習(xí)了數(shù)組的使用,數(shù)組的空間是連續(xù)空間,數(shù)組的大小恒定的,在很多動(dòng)態(tài)數(shù)據(jù)存儲(chǔ)的應(yīng)用場(chǎng)景下,使用不方便;而這篇文章介紹的鏈表結(jié)構(gòu),支持動(dòng)態(tài)增加節(jié)點(diǎn),釋放節(jié)點(diǎn),比較適合存儲(chǔ)動(dòng)態(tài)數(shù)據(jù)的應(yīng)用場(chǎng)景,而且鏈表的空間是存儲(chǔ)在堆上面的,可以動(dòng)態(tài)分配,釋放。從效率上來講,數(shù)組的空間是連續(xù)的,查詢、讀取數(shù)據(jù)數(shù)組占優(yōu)勢(shì);鏈表的優(yōu)勢(shì)在于節(jié)點(diǎn)可以動(dòng)態(tài)增加、動(dòng)態(tài)刪除,刪除支持任意位置的節(jié)點(diǎn)刪除。

特點(diǎn):

數(shù)組的空間是連續(xù)的,可以直接通過[]下標(biāo)訪問。

鏈表的節(jié)點(diǎn)是不連續(xù)的,需要通過每個(gè)節(jié)點(diǎn)的指針,來找到上一個(gè)節(jié)點(diǎn)或者下一個(gè)節(jié)點(diǎn)的地址。

鏈表的每個(gè)節(jié)點(diǎn)就是一個(gè)結(jié)構(gòu)體變量,節(jié)點(diǎn)里有一個(gè)或者兩個(gè)指針,可以保存上一個(gè)節(jié)點(diǎn)和下一個(gè)節(jié)點(diǎn)的地址,方便遍歷鏈表,刪除、插入節(jié)點(diǎn)時(shí)定位位置。

image-20211203094458762

2. 案例: 單向鏈表的創(chuàng)建與使用

下面例子采用函數(shù)封裝的形式編寫,每個(gè)功能都使用子函數(shù)實(shí)現(xiàn)。

image-20211203095248628image-20211203095310341

實(shí)現(xiàn)的功能如下:

  1. 初始化鏈表頭
  2. 插入節(jié)點(diǎn)的函數(shù)(鏈表任意位置插入,鏈表尾插入)
  3. 刪除節(jié)點(diǎn)的函數(shù)(鏈表任意位置刪除、鏈表尾刪除)
  4. 遍歷鏈表,輸出鏈表里的所有信息
#include 
#include 
?
//定義鏈表節(jié)點(diǎn)的結(jié)構(gòu)體
struct app
{
  int a;
  struct app *next; //能保存結(jié)構(gòu)體的地址
};
?
struct app *list_head=NULL; //鏈表的頭指針
?
void list_print(struct app *head);
struct app *list_HeadInit(struct app *head);
void list_add(int a,struct app *head);
void list_del(int a,struct app *head);
?
int main()
{
  //1. 初始化鏈表頭
  list_head=list_HeadInit(list_head);
  //2. 在鏈表尾插入數(shù)據(jù)
  list_add(10,list_head);
  list_add(11,list_head);
  list_add(12,list_head);
  list_add(13,list_head);
  //3. 刪除節(jié)點(diǎn)
  list_del(11,list_head);
  //4. 輸出鏈接節(jié)點(diǎn)里的數(shù)據(jù)
  list_print(list_head);
  return 0;
}
?
/*
函數(shù)功能: 初始化鏈表頭--給鏈表頭申請(qǐng)空間
*/
struct app *list_HeadInit(struct app *head)
{
  if(head==NULL) //沒有空間
   {
    //1. 申請(qǐng)鏈表頭空間
    head=malloc(sizeof(struct app));
    //2. 初始化鏈表頭
    head->next=NULL;
   }
  return head;
}
?
/*
函數(shù)功能: 在鏈表尾插入數(shù)據(jù)
int a  插入的數(shù)據(jù)值
struct app *head  鏈表頭
*/
void list_add(int a,struct app *head)
{
  struct app *new_p=NULL;
  struct app *next_p=head;
  struct app *tmp_p; //保存上一個(gè)節(jié)點(diǎn)的地址
  //1.申請(qǐng)空間并給空間成員賦值
  new_p=malloc(sizeof(struct app));
  new_p->a=a;
  new_p->next=NULL;
?
  //2. 找到鏈表尾
  while(next_p!=NULL)
   {
    tmp_p=next_p;
    next_p=next_p->next; //指針指向下一個(gè)節(jié)點(diǎn)
   }
?
  //3. 插入新節(jié)點(diǎn)--鏈接結(jié)尾
  tmp_p->next=new_p;
}
?
/*
函數(shù)功能: 遍歷輸出鏈接里的所有數(shù)據(jù)
*/
void list_print(struct app *head)
{
  struct app *next_p=head;
  int cnt=0;
  if(head!=NULL)
   {
    while(next_p->next!=NULL)
     {
      next_p=next_p->next;
      printf("鏈表節(jié)點(diǎn)[%d]:a=%d\n",cnt++,next_p->a);
     }
   }
}
?
/*
函數(shù)功能:鏈表的刪除--按照指定的數(shù)據(jù)刪除
*/
void list_del(int a,struct app *head)
{
  struct app *next_p=head;
  struct app *tmp_p; //保存上一個(gè)節(jié)點(diǎn)的地址
  //1. 找到要?jiǎng)h除的鏈表
  if(next_p!=NULL)
   {
    while(next_p->next!=NULL)
     {
      tmp_p=next_p; //保存上一個(gè)節(jié)點(diǎn)的地址
      next_p=next_p->next; //獲取有效節(jié)點(diǎn)的地址
      if(next_p->a==a) //判斷是否需要?jiǎng)h除
       {
        tmp_p->next=next_p->next;
        free(next_p);
       }
     }
   }
}
復(fù)制代碼

3. 案例: 單向循環(huán)鏈表

代碼直接在上面的案例2例子上改造的,區(qū)別就是尾結(jié)點(diǎn)指向了頭結(jié)點(diǎn)而不是NULL。

#include 
#include 
?
//定義鏈表節(jié)點(diǎn)的結(jié)構(gòu)體
struct app
{
  int a;
  struct app *next; //能保存結(jié)構(gòu)體的地址
};
?
struct app *list_head=NULL; //鏈表的頭指針
?
void list_print(struct app *head);
struct app *list_HeadInit(struct app *head);
void list_add(int a,struct app *head);
void list_del(int a,struct app *head);
?
int main()
{
  //1. 初始化鏈表頭
  list_head=list_HeadInit(list_head);
  //2. 在鏈表尾插入數(shù)據(jù)
  list_add(10,list_head);
  list_add(11,list_head);
  list_add(12,list_head);
  list_add(13,list_head);
  //3. 刪除節(jié)點(diǎn)
  list_del(11,list_head);
  //4. 輸出鏈接節(jié)點(diǎn)里的數(shù)據(jù)
  list_print(list_head);
  return 0;
}
?
/*
函數(shù)功能: 初始化鏈表頭--給鏈表頭申請(qǐng)空間
*/
struct app *list_HeadInit(struct app *head)
{
  if(head==NULL) //沒有空間
   {
    //1. 申請(qǐng)鏈表頭空間
    head=malloc(sizeof(struct app));
    //2. 初始化鏈表頭
    head->next=head;
   }
  return head;
}
?
/*
函數(shù)功能: 在鏈表尾插入數(shù)據(jù)
int a  插入的數(shù)據(jù)值
struct app *head  鏈表頭
*/
void list_add(int a,struct app *head)
{
  struct app *new_p=NULL;
  struct app *next_p=head;
  struct app *tmp_p; //保存上一個(gè)節(jié)點(diǎn)的地址
  //1.申請(qǐng)空間并給空間成員賦值
  new_p=malloc(sizeof(struct app));
  new_p->a=a;
  new_p->next=head;
?
  //2. 找到鏈表尾
  if(head!=NULL)
   {
    if(next_p->next==head) //表示第一次插入節(jié)點(diǎn)
     {
      next_p->next=new_p;
      //head ----<節(jié)點(diǎn)1>---head
     }
    else
     {
      while(next_p->next!=head)
       {
        next_p=next_p->next; //指針指向下一個(gè)節(jié)點(diǎn)
       }
      //3. 插入新節(jié)點(diǎn)--鏈接結(jié)尾
      next_p->next=new_p;
     }  
   } 
}
?
/*
函數(shù)功能: 遍歷輸出鏈接里的所有數(shù)據(jù)
*/
void list_print(struct app *head)
{
  struct app *next_p=head;
  int cnt=0;
  if(head!=NULL)
   {
    printf("頭地址: %#x\n",next_p); //頭
    printf("第一個(gè)節(jié)點(diǎn)地址:%#x\n",next_p->next); //下一個(gè)節(jié)點(diǎn)地址
    printf("第二個(gè)節(jié)點(diǎn)地址:%#x\n",next_p->next->next); //下下一個(gè)節(jié)點(diǎn)地址
    printf("第三個(gè)節(jié)點(diǎn)地址:%#x\n",next_p->next->next->next);
    printf("第四個(gè)節(jié)點(diǎn)地址:%#x\n",next_p->next->next->next->next);
  
    while(next_p->next!=head)
     {
      next_p=next_p->next;
      printf("鏈表節(jié)點(diǎn)[%d]:a=%d\n",cnt++,next_p->a);
     }
   }
}
?
/*
函數(shù)功能:鏈表的刪除--按照指定的數(shù)據(jù)刪除
*/
void list_del(int a,struct app *head)
{
  struct app *next_p=head;
  struct app *tmp_p; //保存上一個(gè)節(jié)點(diǎn)的地址
  //1. 找到要?jiǎng)h除的鏈表
  if(next_p!=NULL)
   {
    while(next_p->next!=head)
     {
      tmp_p=next_p; //保存上一個(gè)節(jié)點(diǎn)的地址
      next_p=next_p->next; //獲取有效節(jié)點(diǎn)的地址
      if(next_p->a==a) //判斷是否需要?jiǎng)h除
       {
        tmp_p->next=next_p->next;
        free(next_p);
       }
     }
   }
}
復(fù)制代碼

4. 案例: 創(chuàng)建雙向鏈表循環(huán),實(shí)現(xiàn)插入、刪除、遍歷

雙向鏈表在每個(gè)節(jié)點(diǎn)里新增加了一個(gè)指針,用于保存上一個(gè)節(jié)點(diǎn)的地址,現(xiàn)在的節(jié)點(diǎn)里一個(gè)用兩個(gè)指針,一個(gè)保存上一個(gè)節(jié)點(diǎn)的地址,一個(gè)保存下一個(gè)節(jié)點(diǎn)的地址。

#include 
#include 
?
//定義鏈表節(jié)點(diǎn)的結(jié)構(gòu)體
struct app
{
  int a;
  struct app *next; //下一個(gè)節(jié)點(diǎn)地址
  struct app *prev; //前一個(gè)節(jié)點(diǎn)地址
};
?
//全局變量聲明區(qū)域
struct app *list_head=NULL; //鏈表的頭指針
?
//函數(shù)聲明
struct app *List_HeadInit(struct app *head);
void list_add(int a,struct app *head);
void list_print(struct app *head);
void list_del(int a,struct app *head);
?
int main()
{
  /*1. 初始化鏈表*/
  list_head=List_HeadInit(list_head);
  /*2. 添加鏈表節(jié)點(diǎn)*/
  list_add(10,list_head);
  list_add(11,list_head);
  list_add(12,list_head);
  list_add(13,list_head);
  list_add(14,list_head);
  /*3.刪除指定節(jié)點(diǎn)*/
  list_del(12,list_head);
  /*4. 遍歷輸出所有節(jié)點(diǎn)信息*/
  list_print(list_head);
  return 0;
}
?
/*
函數(shù)功能: 創(chuàng)建鏈表頭
*/
struct app *List_HeadInit(struct app *head)
{
  if(head==NULL)
   {
    head=malloc(sizeof(struct app));
    head->a=0;
    head->next=head;
    head->prev=head;
   }
  return head;
}
?
/*
函數(shù)功能: 添加數(shù)據(jù)-鏈表尾添加數(shù)據(jù)
*/
void list_add(int a,struct app *head)
{
  struct app *next_p=head;
  struct app *new_p=NULL;
  /*1. 申請(qǐng)新的節(jié)點(diǎn)*/
  new_p=malloc(sizeof(struct app));
  new_p->a=a;
  new_p->next=head;
  /*2. 完成新節(jié)點(diǎn)的添加*/
  //遍歷鏈表
  while(next_p->next!=head)
   {
    next_p=next_p->next;
   }
  //添加新節(jié)點(diǎn)
  new_p->prev=next_p;
  next_p->next=new_p;
  //修改鏈表頭的上一個(gè)節(jié)點(diǎn)地址
  head->prev=new_p;
}
?
/*
函數(shù)功能:輸出鏈表里的所有數(shù)據(jù)
*/
void list_print(struct app *head)
{
  struct app *next_p=head;
  int cnt=0;
  /*1. 順向遍歷*/
  while(next_p->next!=head)
   {
    next_p=next_p->next;
    printf("節(jié)點(diǎn)[%d]:%d\n",cnt++,next_p->a);
   }
  /*2. 反向遍歷*/
  next_p=head;
  while(next_p->prev!=head)
   {
    next_p=next_p->prev;
    printf("節(jié)點(diǎn)[%d]:%d\n",cnt--,next_p->a);
   }
}
?
/*
函數(shù)功能:刪除鏈表里的指定節(jié)點(diǎn)
*/
void list_del(int a,struct app *head)
{
  struct app *next_p=head;
  struct app *tmp_p=NULL;
  while(next_p->next!=head)
   {
    tmp_p=next_p; //保存上一個(gè)節(jié)點(diǎn)的地址
    next_p=next_p->next;
    if(next_p->a==a)
     {
      //方式1
      tmp_p->next=tmp_p->next->next;
      tmp_p->next->prev=tmp_p;
?
      //方式2
      //tmp_p->next=next_p->next;
      //next_p->next->prev=tmp_p;
      
      //printf("%d\n",tmp_p->a); //11
      //printf("%d\n",tmp_p->next->a); //13
      //printf("%d\n",next_p->next->prev->a); //11
      free(next_p);
      break;
     }
   }
}

審核編輯 黃昊宇
聲明:本文內(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)投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5083

    文章

    19131

    瀏覽量

    305549
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7605

    瀏覽量

    136934
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C語言實(shí)現(xiàn)動(dòng)態(tài)鏈表的建立

    上期講解了靜態(tài)鏈表的實(shí)例,但是靜態(tài)鏈表建立的節(jié)點(diǎn)數(shù)量有限,畢竟是手工建立,難免也會(huì)出問題, 所以這期講講怎么使用動(dòng)態(tài)的方式建立鏈表,也就是 動(dòng)態(tài)鏈表
    發(fā)表于 01-13 15:16 ?1491次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>實(shí)現(xiàn)動(dòng)態(tài)<b class='flag-5'>鏈表</b>的建立

    C語言算法題:反轉(zhuǎn)一個(gè)單向鏈表

    鏈表是編程學(xué)習(xí)的一個(gè)難點(diǎn)。其實(shí),在C語言編程以及單片機(jī)裸機(jī)開發(fā)中,鏈表運(yùn)用并不多。但是如果想提升嵌入式技能水平或收入水平,可以考慮深入嵌入式系統(tǒng)層面(如參與操作系統(tǒng)設(shè)計(jì)、深入學(xué)習(xí)新的操
    發(fā)表于 06-21 11:07 ?854次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>算法題:反轉(zhuǎn)一個(gè)<b class='flag-5'>單向</b><b class='flag-5'>鏈表</b>

    C語言鏈表知識(shí)點(diǎn)(2)

    C語言鏈表知識(shí)點(diǎn)(2)
    發(fā)表于 08-22 10:38 ?333次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>鏈表</b>知識(shí)點(diǎn)(2)

    C語言單向鏈表

    本帖最后由 snowmay001 于 2016-5-22 15:57 編輯 lianbiao.cpp/* 練習(xí)使用鏈表:創(chuàng)建鏈表、遍歷鏈表、查找節(jié)點(diǎn)、添加節(jié)點(diǎn)、刪除節(jié)點(diǎn)*/#include
    發(fā)表于 05-22 15:53

    C語言鏈表

    C語言鏈表,,,
    發(fā)表于 11-07 17:19

    C語言玩轉(zhuǎn)鏈表

    C語言是必學(xué)的一個(gè)課程,不管你是單片機(jī)還是嵌入式物聯(lián)網(wǎng),都是基礎(chǔ),所以還是要好好學(xué)習(xí)的今天推薦的資料是關(guān)于C語言鏈表的資料我自己看了一下主要
    發(fā)表于 11-13 13:50

    玩轉(zhuǎn)C語言鏈表-鏈表各類操作詳解

    ,它稱為“表尾”,它的地址部分放一個(gè)“NULL”(表示“空地址”),鏈表到此結(jié)束?! ?b class='flag-5'>鏈表的各類操作包括:學(xué)習(xí)單向鏈表的創(chuàng)建、刪除、 插入(無序、有序)、輸出、 排序(選擇、插入、冒泡
    發(fā)表于 09-18 13:30

    如何在C語言中去創(chuàng)建一種雙向鏈表

    雙向鏈表的結(jié)構(gòu)是由哪些部分組成的?如何在C語言中去創(chuàng)建一種雙向鏈表呢?
    發(fā)表于 12-24 06:22

    C語言實(shí)現(xiàn)單鏈表舉例

    所謂鏈表,就是用一組任意的存儲(chǔ)單元存儲(chǔ)線性表元素的一種數(shù)據(jù)結(jié)構(gòu)。鏈表又分為單鏈表、雙向鏈表和循環(huán)鏈表
    發(fā)表于 07-11 16:40 ?87次下載
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>實(shí)現(xiàn)單<b class='flag-5'>鏈表</b>舉例

    C加加建立動(dòng)態(tài)鏈表

    C加加建立動(dòng)態(tài)鏈表利用C語言c++編寫程序
    發(fā)表于 11-19 13:43 ?0次下載

    C語言鏈表相關(guān)資料下載

    C語言鏈表相關(guān)資料
    發(fā)表于 03-08 10:47 ?5次下載

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

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

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

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

    C語言_鏈表總結(jié)

    本篇文章介紹C語言鏈表相關(guān)知識(shí)點(diǎn),涉及鏈表的創(chuàng)建、單向鏈表、循環(huán)
    的頭像 發(fā)表于 08-14 09:53 ?1794次閱讀

    鏈表和雙鏈表的區(qū)別在哪里

    。 上面的三幅圖對(duì)于理解鏈表的插入、刪除很重要,看代碼的時(shí)候要對(duì)著看。 實(shí)際中經(jīng)常使用的一般為帶頭雙向循環(huán)鏈表,下面是一個(gè)雙向循環(huán)鏈表的 d
    的頭像 發(fā)表于 07-27 11:20 ?1681次閱讀
    單<b class='flag-5'>鏈表</b>和雙<b class='flag-5'>鏈表</b>的區(qū)別在哪里