欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言單鏈表的圖文示例講解

 更新時間:2023年02月14日 14:42:05   作者:[Pokemon]大貓貓  
單鏈表是鏈表的其中一種基本結(jié)構(gòu)。一個最簡單的結(jié)點(diǎn)結(jié)構(gòu)如圖所示,它是構(gòu)成單鏈表的基本結(jié)點(diǎn)結(jié)構(gòu)。在結(jié)點(diǎn)中數(shù)據(jù)域用來存儲數(shù)據(jù)元素,指針域用于指向下一個具有相同結(jié)構(gòu)的結(jié)點(diǎn)。?因為只有一個指針結(jié)點(diǎn),稱為單鏈表

在上一篇所講述的 動態(tài)順序表 中存在一些缺陷

1、當(dāng)空間不夠時需要擴(kuò)容,擴(kuò)容是有一定的消耗的

如果每次空間擴(kuò)大一點(diǎn),可能會造成空間的浪費(fèi),而空間擴(kuò)小了,又會造成頻繁的擴(kuò)容2、在順序表中進(jìn)行頭部和中部的插入時需要移動數(shù)據(jù),效率低下

對于順序表的這些缺陷,有如下解決方案

1、需要時申請一塊空間,不需要時將其釋放

2、插入刪除不需要移動數(shù)據(jù)

而鏈表就符合這兩點(diǎn),本篇介紹 無頭單向非循環(huán)鏈表(單鏈表)

一、單鏈表的結(jié)構(gòu)

空鏈表: 此時沒有存儲數(shù)據(jù),只有一個指針指向 NULL

以上便是單鏈表的結(jié)構(gòu):

  • 每一塊空間可以按需申請釋放
  • 插入和刪除不需要移動數(shù)據(jù),修改每塊空間的指針指向即可

在習(xí)慣上將申請的一塊一塊的空間稱為結(jié)點(diǎn),指向第一個結(jié)點(diǎn)的指針稱為頭指針

//數(shù)據(jù)的類型:這里以 int 來舉例
typedef int SLTDataType;
//結(jié)點(diǎn)的類型
typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;
}SLTNode;

二、單鏈表的函數(shù)接口

1. 申請結(jié)點(diǎn)及打印單鏈表

在插入時需要申請結(jié)點(diǎn),為了避免麻煩重復(fù)的操作,這里將申請結(jié)點(diǎn)封裝為一個函數(shù)

申請結(jié)點(diǎn)函數(shù)如下:

SLTNode* BuySLTNode(SLTDataType x)
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	if(newnode == NULL)
	{
		//開辟空間失敗,打印錯誤信息
		perror("malloc");
		//結(jié)束程序
		exit(-1);
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

為了驗證插入、刪除等得到的結(jié)果是否正確,提供打印單鏈表的函數(shù),這里數(shù)據(jù)類型以 int 為例,當(dāng)讀者采用的類型不同時,自行更改函數(shù)即可

打印單鏈表函數(shù)如下:

void SLTPrint(SLTNode* phead)
{
	SLTNode* cur = phead;
	//打印數(shù)據(jù)
	while(cur)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");
}

2. 尾插尾刪

尾插:在鏈表的最后一個結(jié)點(diǎn)之后插入結(jié)點(diǎn)

尾插函數(shù)如下:

//在鏈表為空時,需要改變頭指針,這里采用傳二級指針的方式
void SLTPushBack(SLTNode** pphead, SLTDataType x)
{
	//申請結(jié)點(diǎn)
	SLTNode* newnode = BuySLTNode(x);
	//鏈表為空時
	if(*pphead == NULL)
	{
		*pphead = newnode;
	}
	else
	{
		//找到最后一個結(jié)點(diǎn)
		SLTNode* ptail = *pphead;
		while(ptail->next)
		{
			ptail = ptail->next;
		}
		ptail->next = newnode;
	}
}

尾刪:刪除鏈表最后一個結(jié)點(diǎn)

尾刪函數(shù)如下:

//鏈表只有一個結(jié)點(diǎn)時,需要改變頭指針,這里采用傳二級指針的方式
void SLTPopBack(SLTNode** pphead)
{
	//鏈表為空時,無法刪除
	assert(*pphead);
	//鏈表只有一個結(jié)點(diǎn)時
	if((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		//找到倒數(shù)第二個結(jié)點(diǎn)
		SLTNode* ptail = *pphead;
		while(ptail->next->next)
		{
			ptail = ptail->next;
		}
		free(ptail->next);
		ptail->next = NULL;
	}
}

3. 頭插頭刪

頭插: 在第一個結(jié)點(diǎn)之前插入新結(jié)點(diǎn)

頭插函數(shù)如下:

//需要改變頭指針,這里采用傳二級指針的方式
void SLTPushFront(SLTNode** pphead, SLTDataType x)
{
	//申請結(jié)點(diǎn)
	SLTNode* newnode = BuySLTNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

頭刪:刪除鏈表的第一個結(jié)點(diǎn)

頭刪函數(shù)如下:

//需要改變頭指針,這里采用傳二級指針的方式
void SLTPopFront(SLTNode** pphead)
{
	//鏈表為空時,無法刪除
	assert(*pphead);
	//保存第二個結(jié)點(diǎn)
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	*pphead = next;
}

4. 中間插入和刪除

中間插入:通過后面介紹的查找函數(shù) SLTFind 獲得指向結(jié)點(diǎn)的指針 pos,在 pos 指向的 結(jié)點(diǎn)之前 或 之后 插入結(jié)點(diǎn)

1. 在 pos 指向的結(jié)點(diǎn)之后插入結(jié)點(diǎn)

在 pos 之后插入結(jié)點(diǎn)函數(shù)如下:

void SLTInsertAfter(SLTNode* pos, SLTDataType x)
{
	//pos 不能為空
	assert(pos);
	//申請結(jié)點(diǎn)
	SLTNode* newnode = BuySLTNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}

2. 在 pos 指向的結(jié)點(diǎn)之前插入結(jié)點(diǎn)

在 pos 之前插入結(jié)點(diǎn)函數(shù)如下:

//pos 指向頭結(jié)點(diǎn)時,需要改變頭指針,這里采用傳二級指針的方式
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	//pos 不能為空
	assert(pos);
	//頭插
	if(*pphead == pos)
	{
		SLTPushFront(pphead, x);
	}
	else
	{
		//找到 pos 的前一個結(jié)點(diǎn)
		SLTNode* prev = *pphead;
		while(prev->next != pos)
		{
			prev = prev->next;
		}
		//申請結(jié)點(diǎn)
		SLTNode* newnode = BuySLTNode(x);
		newnode->next = pos;
		prev->next = newnode;
	}
}

中間刪除:通過后面介紹的查找函數(shù) SLTFind 獲得指向結(jié)點(diǎn)的指針 pos,刪除 pos 指向的結(jié)點(diǎn) 或 后一個結(jié)點(diǎn)

3. 刪除 pos 指向的結(jié)點(diǎn)的后一個結(jié)點(diǎn)

刪除 pos 之后的結(jié)點(diǎn)函數(shù)如下:

void SLTEraseAfter(SLTNode* pos)
{
	//pos 不能為空
	assert(pos);
	//指向最后一個結(jié)點(diǎn)時,不做處理
	if(pos->next == NULL)
	{
		return;
	}
	else
	{
		//保存后一個結(jié)點(diǎn)
		SLTNode* next = pos->next;
		pos->next = next->next;
		free(next);
	}
}

4. 刪除 pos 指向的結(jié)點(diǎn)

刪除 pos 指向的結(jié)點(diǎn)函數(shù)如下:

//pos 指向頭結(jié)點(diǎn)時,需要改變頭指針,這里采用傳二級指針的方式
void SLTErase(SLTNode** pphead, SLTNode* pos)
{
	//pos 不能為空
	assert(pos);
	//頭刪
	if (*pphead == pos)
	{
		SLTPopFront(pphead);
	}
	else
	{
		//找到 pos 的前一個結(jié)點(diǎn)
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
	}
}

6. 查找

查找:如果數(shù)據(jù)存在,返回該數(shù)據(jù)結(jié)點(diǎn)的指針,不存在返回 NULL

查找函數(shù)如下:

SLTNode* SLTFind(SLTNode* phead, SLTDataType x)
{
	SLTNode* cur = phead;
	//查找
	while (cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

7. 銷毀單鏈表

在單鏈表中,存儲數(shù)據(jù)的結(jié)點(diǎn)是由自己開辟的,當(dāng)不使用單鏈表時,應(yīng)將其銷毀

銷毀單鏈表函數(shù)如下:

需要將頭指針置空,這里采用傳二級指針的方式
void SLTDestroy(SLTNode** pphead)
{
	SLTNode* cur = *pphead;
	while (cur)
	{
		//保存下一個結(jié)點(diǎn)
		SLTNode* nextnode = cur->next;
		free(cur);
		cur = nextnode;
	}
	//將頭指針置空
	*pphead = NULL;
}

到此這篇關(guān)于C語言單鏈表的圖文示例講解的文章就介紹到這了,更多相關(guān)C語言單鏈表 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++菱形繼承和虛繼承的實現(xiàn)

    C++菱形繼承和虛繼承的實現(xiàn)

    本文主要介紹了C++菱形繼承和虛繼承的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • C++中BitBlt的使用方法詳解

    C++中BitBlt的使用方法詳解

    這篇文章主要介紹了C++中BitBlt的使用方法詳解的相關(guān)資料,希望通過本文能幫助到大家,需要的朋友可以參考下
    2017-09-09
  • 深入理解雙指針的兩種用法

    深入理解雙指針的兩種用法

    本篇文章是對雙指針的兩種用法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C語言實現(xiàn)掃雷OvO(完整代碼)

    C語言實現(xiàn)掃雷OvO(完整代碼)

    相信大家都玩過掃雷游戲,因為它太經(jīng)典了,今天我們用C語言來模擬實現(xiàn)掃雷游戲,結(jié)合示例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧
    2022-04-04
  • FFmpeg實現(xiàn)音頻漸響效果參數(shù)值詳解

    FFmpeg實現(xiàn)音頻漸響效果參數(shù)值詳解

    這篇文章主要為大家介紹了FFmpeg實現(xiàn)音頻漸響效果參數(shù)值詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • 關(guān)于C++中虛擬繼承的一些總結(jié)分析

    關(guān)于C++中虛擬繼承的一些總結(jié)分析

    虛擬繼承在一般的應(yīng)用中很少用到,所以也往往被忽視,這也主要是因為在C++中,多重繼承是不推薦的,也并不常用
    2013-09-09
  • 詳解C++ thread用法總結(jié)

    詳解C++ thread用法總結(jié)

    這篇文章主要介紹了詳解C++ thread用法總結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • exit和atexit的區(qū)別詳細(xì)解析

    exit和atexit的區(qū)別詳細(xì)解析

    以下是對exit與atexit的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下
    2013-09-09
  • C++實現(xiàn)LeetCode(33.在旋轉(zhuǎn)有序數(shù)組中搜索)

    C++實現(xiàn)LeetCode(33.在旋轉(zhuǎn)有序數(shù)組中搜索)

    這篇文章主要介紹了C++實現(xiàn)LeetCode(33.在旋轉(zhuǎn)有序數(shù)組中搜索),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C++ 函數(shù)重載詳情介紹

    C++ 函數(shù)重載詳情介紹

    這篇文章主要介紹了C++ 函數(shù)重載詳情,函數(shù)重載還有一個別名叫函數(shù)多態(tài),函數(shù)多態(tài)是C++在C語言基礎(chǔ)上的新特性,它可以讓我們使用多個同名函數(shù),下面來看看文章具體內(nèi)容的介紹
    2021-11-11

最新評論