C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之單鏈表的實(shí)現(xiàn)
一.為什么使用鏈表
在學(xué)習(xí)鏈表以前,我們存儲(chǔ)數(shù)據(jù)用的方式就是數(shù)組。使用數(shù)組的好處就是便于查找數(shù)據(jù),但缺點(diǎn)也很明顯。
使用前需聲明數(shù)組的長(zhǎng)度,一旦聲明長(zhǎng)度就不能更改
插入和刪除操作需要移動(dòng)大量的數(shù)組元素,效率慢
只能存儲(chǔ)一種類(lèi)型的數(shù)據(jù).
為了解決上述的問(wèn)題,我們就可以使用鏈表來(lái)存儲(chǔ)數(shù)據(jù)。
二.鏈表的概念
概念:鏈表是一種物理存儲(chǔ)結(jié)構(gòu)上非連續(xù)、非順序的存儲(chǔ)結(jié)構(gòu),數(shù)據(jù)元素的邏輯順序是通過(guò)鏈表中的指針鏈接次序?qū)崿F(xiàn)的
三.鏈表的實(shí)現(xiàn)
3.1 創(chuàng)建鏈表前須知
結(jié)點(diǎn):鏈表中每一個(gè)元素稱(chēng)為“結(jié)點(diǎn)”,每個(gè)結(jié)點(diǎn)都應(yīng)包括兩個(gè)部分:一為用戶(hù)需要用的實(shí)際數(shù)據(jù);二為下一個(gè)結(jié)點(diǎn)的地址,
頭結(jié)點(diǎn):在單鏈表的第一個(gè)結(jié)點(diǎn)之前附設(shè)一個(gè)結(jié)點(diǎn),這個(gè)節(jié)點(diǎn)不存儲(chǔ)數(shù)據(jù),稱(chēng)之為頭結(jié)點(diǎn)
3.2 定義結(jié)構(gòu)體
#include <stdio.h> #include <stdlib.h> #include <assert.h> typedef int SLDateType; //鏈表中存儲(chǔ)的數(shù)據(jù)類(lèi)型,可換成其他 typedef struct SListNode { SLDateType date; struct SListNode* next; //指向下一個(gè)節(jié)點(diǎn)的指針 }SListNode;
3.3 申請(qǐng)一個(gè)節(jié)點(diǎn)
SListNode* BuyListNode(SLDateType x) { SListNode* newNode = (SListNode*)malloc(sizeof(SListNode)); if (NULL == newNode) { printf("malloc error\n"); //內(nèi)存開(kāi)辟失敗 exit(-1); } else { newNode->date = x; // 給新節(jié)點(diǎn)賦值 newNode->next = NULL; } return newNode; }
3.4 鏈表的頭插
void SListPushFront(SListNode** pphead/*要改動(dòng)頭指針,所以要傳遞二級(jí)指針*/, SLDateType x) { SListNode* newNode = BuyListNode(x); //申請(qǐng)節(jié)點(diǎn) newNode->next = *pphead; *pphead = newNode; }
3.5 鏈表的尾插
void SListPushBack(SListNode** pphead, SLDateType x) { SListNode* newNode = BuyListNode(x); if (*pphead == NULL) //若頭指針為空,則鏈表為空鏈表,直接將新節(jié)點(diǎn)接到頭指針后 { *pphead = newNode; } else { SListNode* tail = *pphead; while (tail->next != NULL) //找鏈表的尾部 { tail = tail->next; } tail->next = newNode;//將新節(jié)點(diǎn)接到尾部 } }
3.6 鏈表的尾刪
void SListPopBack(SListNode** pphead) { assert(pphead); if (*pphead == NULL)//鏈表為空,則不進(jìn)行任何操作 { return; } else if ((*pphead)->next == NULL) //鏈表只有一個(gè)節(jié)點(diǎn) { free(*pphead); *pphead = NULL; } else//其余情況 { SListNode* tail = *pphead; //鏈表的尾部節(jié)點(diǎn) SListNode* pre = NULL;//鏈表尾的前一個(gè)節(jié)點(diǎn) while (tail->next != NULL)//找尾 { pre = tail; tail = tail->next; } pre->next = tail->next; //將尾節(jié)點(diǎn)的指針域賦值給前一個(gè)節(jié)點(diǎn)的指針域 free(tail); } }
3.7 鏈表的頭刪
void SListPopFront(SListNode** pphead) { assert(pphead); if (*pphead == NULL) //鏈表為空什么也不做 { return; } else { SListNode* head = *pphead;//記錄原本的第一個(gè)節(jié)點(diǎn) *pphead = head->next; //讓頭指針指向第二個(gè)節(jié)點(diǎn) free(head);//釋放第一個(gè)節(jié)點(diǎn) } }
3.8 尋找某節(jié)點(diǎn)
SListNode* SListFind(SListNode* phead, SLDateType x) { SListNode* cur = phead; while (cur != NULL) { if (cur->date == x) //找到則返回該節(jié)點(diǎn) { return cur; } cur = cur->next; } return NULL; //未找到則返回空 }
3.9 在指定節(jié)點(diǎn)前插入節(jié)點(diǎn)
void SListInsert(SListNode** pphead, SListNode* pos/*要插入的位置*/, SLDateType x) { assert(pphead); assert(pos); if (*pphead == pos) { SListPushFront(pphead, x); } else { SListNode* cur = *pphead; //當(dāng)前所指向的位置 SListNode* pre = NULL; //前一個(gè)節(jié)點(diǎn) while (cur != pos) { pre = cur; cur = cur->next; } SListNode* newNode = BuyListNode(x); pre->next = newNode; newNode->next = cur; } }
3.10 刪除指定節(jié)點(diǎn)前的節(jié)點(diǎn)
void SListErase(SListNode** pphead, SListNode* pos/*要插入的位置*/) { assert(pphead); assert(pos); if (*pphead == pos) { SListPopFront(pphead); } else { SListNode* cur = *pphead; SListNode* pre = *pphead; while (cur != pos) { pre = cur; cur = cur->next; } pre->next = cur->next; free(cur); } }
3.11 鏈表的銷(xiāo)毀
void SListDestory(SListNode** pphead) { if (*pphead == NULL) { return; } else { while (*pphead != NULL) { SListNode* cur = *pphead; *pphead = cur->next; free(cur); } } }
到此這篇關(guān)于C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之單鏈表的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)C語(yǔ)言 單鏈表內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之判斷循環(huán)鏈表空與滿(mǎn)
- C語(yǔ)言如何建立動(dòng)態(tài)鏈表問(wèn)題
- C語(yǔ)言利用鏈表實(shí)現(xiàn)學(xué)生成績(jī)管理系統(tǒng)
- C語(yǔ)言中單鏈表(不帶頭結(jié)點(diǎn))基本操作的實(shí)現(xiàn)詳解
- C語(yǔ)言實(shí)現(xiàn)手寫(xiě)Map(數(shù)組+鏈表+紅黑樹(shù))的示例代碼
- C語(yǔ)言中如何實(shí)現(xiàn)單鏈表刪除指定結(jié)點(diǎn)
- C語(yǔ)言刷題判斷鏈表中是否有環(huán)題解
相關(guān)文章
Win32應(yīng)用程序(SDK)設(shè)計(jì)原理詳解
這篇文章主要介紹了Win32應(yīng)用程序(SDK)設(shè)計(jì)原理,對(duì)于理解win32應(yīng)用程序運(yùn)行原理有很大的幫助,需要的朋友可以參考下2014-08-08C語(yǔ)言示例講解while循環(huán)語(yǔ)句的用法
在不少實(shí)際問(wèn)題中有許多具有規(guī)律性的重復(fù)操作,因此在程序中就需要重復(fù)執(zhí)行某些語(yǔ)句。一組被重復(fù)執(zhí)行的語(yǔ)句稱(chēng)之為循環(huán)體,C語(yǔ)言while語(yǔ)句可以是單個(gè)語(yǔ)句,也可以是一個(gè)語(yǔ)句塊,其條件可以是任意表達(dá)式,true是任意非零值,當(dāng)條件為真時(shí),循環(huán)進(jìn)行迭代2022-06-06C語(yǔ)言中QString與QByteArray互相轉(zhuǎn)換的方法
本文主要介紹了C語(yǔ)言中QString與QByteArray互相轉(zhuǎn)換的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05C++實(shí)現(xiàn)正態(tài)隨機(jī)分布的方法
本篇介紹了,使用c++實(shí)現(xiàn)正態(tài)隨機(jī)分布的實(shí)現(xiàn)方法。需要的朋友參考下2013-05-05深入分析為Visual Assist設(shè)置快捷鍵的方法
本篇文章是對(duì)為Visual Assist設(shè)置快捷鍵的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05OpenCV4.1.0+VisualStudio2019開(kāi)發(fā)環(huán)境搭建(超級(jí)簡(jiǎn)單)
這篇文章主要介紹了OpenCV4.1.0+VisualStudio2019開(kāi)發(fā)環(huán)境搭建(超級(jí)簡(jiǎn)單),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03C++基于控制臺(tái)實(shí)現(xiàn)的貪吃蛇小游戲
這篇文章主要介紹了C++基于控制臺(tái)實(shí)現(xiàn)的貪吃蛇小游戲,實(shí)例分析了貪吃蛇游戲的原理與C++實(shí)現(xiàn)技巧,是非常經(jīng)典的游戲算法,需要的朋友可以參考下2015-04-04C語(yǔ)言中函數(shù)指針與軟件設(shè)計(jì)經(jīng)驗(yàn)總結(jié)
今天小編就為大家分享一篇關(guān)于C語(yǔ)言中函數(shù)指針與軟件設(shè)計(jì)經(jīng)驗(yàn)總結(jié),小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-12-12