数据结构之单链表

前言

顺序表中虽然代码简洁但是存在许多问题

  1. 中间/头部的插入删除,时间复杂度较高(O(N))
  2. 增容时需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。
  3. 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再继续插入了5个数据,后面没有数据插入了,那么就浪费了95个数据空间。

如何解决上述问题呢?这里我们引入链表

链表的概念及结构

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

如下图
在这里插入图片描述
注意:

  1. 从上图可看出,链式结构在逻辑上是连续的,但是在物理上不一定连续
  2. 现实中的结点一般都是从堆上申请出来的
  3. 从堆上申请的空间,是按照一定的策略来分配的,两次申请的空间可能连续,也可能不连续

顺序表的功能实现

先创建三个文件SList.h(头文件),SList.c(函数文件),test.c(测试文件)
在这里插入图片描述

创建结点

typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;			//存放该结点的数据
	struct SListNode* next;		//存放下一结点的地址
}SLTNode;

单链表打印

将所有结点的数据按顺序打印出来

void SListPrint(SLTNode* phead)
{
	SLTNode* cur = phead;
	while (cur != NULL)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");
}

单链表尾插

思路

  1. 遍历找到尾结点
  2. 创建新的结点
  3. 将新的结点连接到尾结点上
SLTNode* BuySListNode(SLTDataType x)
{
	//开辟新结点
	SLTNode* node = (SLTNode*)malloc(sizeof(SLTNode));
	if (node == NULL)
	{
		printf("malloc fail\n");
		exit(-1);
	}
	//对新结点进行初始化
	node->data = x;
	node->next = NULL;

	return node;
}

void SListPushBack(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	if (*pphead == NULL)
	{
		SLTNode* newnode = BuySListNode(x);
		*pphead = newnode;
	}
	else
	{
		SLTNode* tail = *pphead;
		//找尾结点
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		SLTNode* newnode = BuySListNode(x);
		//连接
		tail->next = newnode;
	}
}

注意:这里可能会对首位结点进行修改需要传二级指针,如果传一级指针,在函数中修改首位结点无法返回给主程序(类似传址和传值问题)

检测
在这里插入图片描述

单链表头插

思路

  1. 创建一个新结点
  2. 将新结点插入链表头部
void SListPushFront(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	//创建一个新结点
	SLTNode* newnode = BuySListNode(x);
	//头插
	newnode->next = *pphead;
	*pphead = newnode;
}

检测
在这里插入图片描述

单链表尾删

思路很简单找到尾结点释放就行了

void SListPopBack(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//单个结点
	if ((*pphead)->next == NULL)//->优先级比*高需要(*pphead)
	{
		free(*pphead);
		*pphead = NULL;
	}
	//多个结点
	else
	{
		SLTNode* prev = NULL;
		SLTNode* tail = *pphead;
		//找尾结点
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		//释放
		free(tail);
		prev->next = NULL;
	}
}

检测
在这里插入图片描述

单链表头删

将头结点删除释放,并把第二个结点置为头结点

void SListPopFront(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	(*pphead) = next;
}

检测
在这里插入图片描述

计算链表大小

遍历,计算结点个数

int SListSize(SLTNode* phead)
{
	int size = 0;
	SLTNode* cur = phead;
	while (cur)
	{
		size++;
		cur = cur->next;
	}
	return size;
}

检测
在这里插入图片描述

判断链表是否为空

这里需要引入一个新类型bool,如果该函数为真返回1,为假返回0,需要引入头文件<stdbool.h>

bool SListEmpty(SLTNode* phead)
{
	return phead == NULL;
}

查找某个值

思路

  1. 遍历链表
  2. 将结点的值和要找的值进行比较,相同返回该结点的坐标
  3. 如果没找到返回空指针
SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{
	SLTNode* cur = phead;
	while (cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		else
		{
			cur = cur->next;
		}
	}
	return NULL;
}

在指定位置插入

思路

  1. 找到目标结点的前一个结点
  2. 新建一个结点保存数据
  3. 将新结点连如链表中

如下图
在这里插入图片描述

void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(pphead);
	assert(pos);
	//如果在第一个结点前插入就首插
	if (*pphead == pos)
	{
		SListPushFront(pphead, x);
	}
	else
	{
		//找到pos前一个结点
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		SLTNode* newnode = BuySListNode(x);
		newnode->next = pos;
		prev->next = newnode;
	}
}

检测
在这里插入图片描述

指定位置删除

思路

  1. 找到目标结点的前一个结点,和后一个结点
  2. 将目标结点的前后结点相连
  3. 删除目标结点

在这里插入图片描述

void SListErase(SLTNode** pphead, SLTNode* pos)
{
	assert(pphead);
	assert(pos);
	//一个结点时相当于头删
	if (*pphead == pos)
	{
		SListPopFront(pphead);
	}
	else
	{
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
		pos = NULL;
	}
}

检测
在这里插入图片描述

单链表的销毁

将单链表依次销毁

void SListDestory(SLTNode** plist)
{
	assert(plist);
	SLTNode* cur = *plist;
	//将单链表依次销毁
	while (cur)
	{
		SLTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	*plist = NULL;
}

检测
在这里插入图片描述

总结

SList.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include <stdbool.h>

typedef int SLTDataType;

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

void SListPrint(SLTNode* phead);
//尾插
void SListPushBack(SLTNode** pphead, SLTDataType x);
//头插
void SListPushFront(SLTNode** pphead, SLTDataType x);
//尾删
void SListPopBack(SLTNode** pphead);
//头删
void SListPopFront(SLTNode** pphead);
//计算链表大小
int SListSize(SLTNode* phead);
//判断链表是否为空
bool SListEmpty(SLTNode* phead);
//查找
SLTNode* SListFind(SLTNode* phead, SLTDataType x);
//pos位置前插
void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x);
// 在pos位置后面去插入x
void SListInsertAfter(SLTNode* pos, SLTDataType x);
// 删除pos位置的值
void SListErase(SLTNode** pphead, SLTNode* pos);
//删除pos后的值
void SListEraseAfter(SLTNode* pos);
// 单链表的销毁
void SListDestory(SLTNode** plist);

SList.c

#include"SList.h"
void SListPrint(SLTNode* phead)
{
	SLTNode* cur = phead;
	while (cur != NULL)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");
}

SLTNode* BuySListNode(SLTDataType x)
{
	//开辟新结点
	SLTNode* node = (SLTNode*)malloc(sizeof(SLTNode));
	if (node == NULL)
	{
		printf("malloc fail\n");
		exit(-1);
	}
	//对新结点进行初始化
	node->data = x;
	node->next = NULL;

	return node;
}

void SListPushBack(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	if (*pphead == NULL)
	{
		SLTNode* newnode = BuySListNode(x);
		*pphead = newnode;
	}
	else
	{
		SLTNode* tail = *pphead;
		//找尾结点
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		SLTNode* newnode = BuySListNode(x);
		//连接
		tail->next = newnode;
	}
}

void SListPushFront(SLTNode** pphead, SLTDataType x)
{
	assert(pphead);
	//创建一个新结点
	SLTNode* newnode = BuySListNode(x);
	//头插
	newnode->next = *pphead;
	*pphead = newnode;
}

void SListPopBack(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//单个结点
	if ((*pphead)->next == NULL)//->优先级比*高需要(*pphead)
	{
		free(*pphead);
		*pphead = NULL;
	}
	//多个结点
	else
	{
		SLTNode* prev = NULL;
		SLTNode* tail = *pphead;
		//找尾结点
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		//释放
		free(tail);
		prev->next = NULL;
	}
}

void SListPopFront(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	(*pphead) = next;
}

int SListSize(SLTNode* phead)
{
	int size = 0;
	SLTNode* cur = phead;
	while (cur)
	{
		size++;
		cur = cur->next;
	}
	return size;
}

bool SListEmpty(SLTNode* phead)
{
	return phead == NULL;
}

SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{
	SLTNode* cur = phead;
	while (cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		else
		{
			cur = cur->next;
		}
	}
	return NULL;
}

void SListInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(pphead);
	assert(pos);
	//如果在第一个结点前插入就首插
	if (*pphead == pos)
	{
		SListPushFront(pphead, x);
	}
	else
	{
		//找到pos前一个结点
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		SLTNode* newnode = BuySListNode(x);
		newnode->next = pos;
		prev->next = newnode;
	}
}

void SListInsertAfter(SLTNode* pos, SLTDataType x)
{
	assert(pos);
	SLTNode* newnode = BuySListNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}

void SListErase(SLTNode** pphead, SLTNode* pos)
{
	assert(pphead);
	assert(pos);
	//一个结点时相当于头删
	if (*pphead == pos)
	{
		SListPopFront(pphead);
	}
	else
	{
		SLTNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
		pos = NULL;
	}
}


void SListEraseAfter(SLTNode* pos)
{
	assert(pos);
	assert(pos->next);
	SLTNode* next = pos->next;
	pos->next = next->next;
	free(next);
	next = NULL;
}

void SListDestory(SLTNode** plist)
{
	assert(plist);
	SLTNode* cur = *plist;
	//将单链表依次销毁
	while (cur)
	{
		SLTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	*plist = NULL;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值