数据结构3—单链表(附源码)

1.概念与结构

概念:链表是一种物理存储结构上非连续 、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。

结构:由一个个结点构成,每个结点是一个结构体(里面包含了数据域和指针域,其中指针域存储的是下一个结点的地址)

2.源代码

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>

//定义链表(结点)的结构

typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;
}SLTNode;

void SLTPrint(SLTNode* phead);//打印链表

SLTNode* SLTBuyNode(SLTDataType x);//创建结点

void SLTPushBack(SLTNode** pphead,SLTDataType x);//尾插法

void SLTPushFront(SLTNode** pphead, SLTDataType x);//头插法

void SLTPopBack(SLTNode** pphead);//尾删法

void SLTPopFront(SLTNode** pphead);//头删法

SLTNode* SLTFind(SLTNode* phead, SLTDataType x);//查找

void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x);//在pos前插入数据

void SLTInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x);//在pos后插入数据

void SLTErase(SLTNode** pphead, SLTNode* pos);//删除pos结点

void SLTEraseAfter(SLTNode* pos);//删除pos的下一个结点

void SListDestroy(SLTNode** pphead);//销毁链表
cpp 复制代码
void SLTPrint(SLTNode* phead)
{
	SLTNode* pcur = phead;
	while (pcur)
	{
		printf("%d->", pcur->data);
		pcur = pcur->next;
	}
	printf("NULL");
}
cpp 复制代码
SLTNode* SLTBuyNode(SLTDataType x)
{
	SLTNode* Node = (SLTNode*)malloc(sizeof(SLTNode));
	if (Node == NULL)
	{
		perror("malloc fail!");
		exit(1);
	}
	Node->data = x;
	Node->next = NULL;

	return Node;
}
cpp 复制代码
void SLTPushBack(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	SLTNode* NewNode=SLTBuyNode(x);
	if (*pphead == NULL)
	{
		*pphead = NewNode;
	}
	else
	{
		SLTNode* pcur = *pphead;
		while (pcur->next)
		{
			pcur = pcur->next;
		}
		pcur->next = NewNode;
	}
}
cpp 复制代码
void SLTPushFront(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	SLTNode* NewNode = SLTBuyNode(x);
		NewNode->next = *pphead;
		*pphead = NewNode;
}
cpp 复制代码
void SLTPopBack(SLTNode** pphead)
{
	assert(pphead && *pphead);

	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SLTNode* ptail = *pphead;
		SLTNode* prev = *pphead;
		while (ptail->next)
		{
			prev = ptail;
			ptail = ptail->next;
		}
		prev->next = NULL;
		free(ptail);
		ptail->next = NULL;
	}
}
cpp 复制代码
void SLTPopFront(SLTNode** pphead)
{
	assert(pphead && *pphead);
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SLTNode* next = (*pphead)->next;
		free(*pphead);
		*pphead = next;
	}
}
cpp 复制代码
SLTNode* SLTFind(SLTNode* phead, SLTDataType x)
{
	assert(phead);
	SLTNode* pcur = phead;

	int count = 1;
	while (pcur)
	{
		if (pcur->data == x)
		{
			return pcur;
		}
		else 
		{
			pcur = pcur->next;
		}
	}
	return NULL;
}
cpp 复制代码
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(pphead);
	assert(pos);
	SLTNode* InsertData=SLTBuyNode(x);
	SLTNode* prev = *pphead;

	if (pos == *pphead)
	{
		SLTPushFront(pphead,x);
	}
	else 
	{
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		InsertData->next = pos;
		prev->next = InsertData;
	}
}
cpp 复制代码
void SLTInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(pphead);
	assert(pos);
	SLTNode* InsertData = SLTBuyNode(x);
	SLTNode* prev = *pphead;
	if (pos == *pphead)
	{
		SLTPushFront(pphead, x);
	}
	else
	{
		while (prev != pos)
		{
			prev = prev->next;
		}
		prev = prev->next;

		pos->next = InsertData;
		InsertData->next = prev;
	}
}
cpp 复制代码
void SLTErase(SLTNode** pphead, SLTNode* pos)
{
	assert(pphead && *pphead);
	assert(pos);
	SLTNode* prev = *pphead;

	if (pos == *pphead)
	{
		SLTPopFront(pphead);
	}
	else
	{
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = prev->next->next;
		free(pos);
		pos = NULL;
	}
}
cpp 复制代码
void SLTEraseAfter(SLTNode* pos)
{
	assert(pos);
	SLTNode* del = pos->next;
	pos->next= del->next;
	free(del);
	del = NULL;
}
cpp 复制代码
void SListDestroy(SLTNode** pphead)
{
	assert(*pphead && pphead);
	SLTNode* prev = *pphead;
	while (prev)
	{
		SLTNode* next= prev->next;
		free(prev);
		prev=next;
	}
	*pphead = NULL;
}

3.测试部分

cpp 复制代码
void CreatSList()
{
	//链表是由一个个的结点构成
	SLTNode* node1 = (SLTNode*)malloc(sizeof(SLTNode));
	node1->data = 1;

	SLTNode* node2 = (SLTNode*)malloc(sizeof(SLTNode));
	node2->data = 2;

	SLTNode* node3 = (SLTNode*)malloc(sizeof(SLTNode));
	node3->data = 3;

	SLTNode* node4 = (SLTNode*)malloc(sizeof(SLTNode));
	node4->data = 4;

	node1->next = node2;
	node2->next = node3;
	node3->next = node4;
	node4->next = NULL;

	SLTNode* pList = node1;

	SLTPushBack(&pList,5);
	SLTPushFront(&pList, 0);
	//SLTPopBack(&pList);
	//SLTPopFront(&pList);
	SLTInsert(&pList,node3,11);
	SLTInsertAfter(&pList, node3, 22);
	//查找
	SLTNode* find = SLTFind(pList, 3);
	SLTErase(&pList, find);
	//打印链表
	SLTPrint(pList);
	SListDestroy(&pList);
}




int main()
{
	CreatSList();

	return 0;
}
相关推荐
一棵开花的树,枝芽无限靠近你8 分钟前
【PPTist】幻灯片放映
前端·笔记·学习·编辑器·pptist
robin_suli11 分钟前
穷举vs暴搜vs深搜vs回溯vs剪枝系列一>优美的排列
算法·剪枝·深度优先遍历·回溯
索然无味io30 分钟前
应急响应之入侵排查(下)
linux·windows·笔记·学习·网络安全·安全威胁分析
比特在路上35 分钟前
OJ12:160. 相交链表
c语言·数据结构·算法·链表·1024程序员节
隼玉38 分钟前
【STM32-学习笔记-8-】I2C通信
c语言·笔记·stm32·学习
在线OJ的阿川43 分钟前
【OJ刷题】同向双指针问题3
c++·学习·算法
比特在路上44 分钟前
初阶数据结构【队列及其接口的实现】
c语言·开发语言·数据结构
多多*1 小时前
JUC Java并发编程 高级 学习大纲 动员
java·开发语言·学习·面试·架构·bash·intellij-idea
surtr11 小时前
【高阶数据结构】线段树加乘(维护序列)详细解释乘与加懒标记
数据结构·c++·算法
努力的CV战士1 小时前
C语言-数据结构-队列
c语言·链表·队列