数据结构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;
}
相关推荐
柠檬少少开发19 分钟前
图像拼接算法及实现(一)
人工智能·算法·计算机视觉
DreamByte23 分钟前
Python Tkinter小程序
开发语言·python·小程序
jnrjian28 分钟前
USE_CONCAT in list OR 以及 filter Nest LOOP
数据结构·list
覆水难收呀31 分钟前
三、(JS)JS中常见的表单事件
开发语言·前端·javascript
阿华的代码王国35 分钟前
【JavaEE】多线程编程引入——认识Thread类
java·开发语言·数据结构·mysql·java-ee
繁依Fanyi41 分钟前
828 华为云征文|华为 Flexus 云服务器部署 RustDesk Server,打造自己的远程桌面服务器
运维·服务器·开发语言·人工智能·pytorch·华为·华为云
weixin_486681141 小时前
C++系列-STL容器中统计算法count, count_if
开发语言·c++·算法
基德爆肝c语言1 小时前
C++入门
开发语言·c++
LN花开富贵1 小时前
stm32g431rbt6芯片中VREF+是什么?在电路中怎么设计?
笔记·stm32·单片机·嵌入式硬件·学习
怀九日1 小时前
C++(学习)2024.9.18
开发语言·c++·学习·面向对象·引用·