C语言数据结构单链表接口函数全面讲解教程

目录
  • 前言
  • 一、链表的概念及结构
    • 1.概念
  • 二、链表的使用
    • 1.遍历整个链表
    • 2.尾插
    • 3.头插
    • 4.头删
    • 5.尾删
    • 6.任意位置插入数据
    • 7.任意位置删除数据
  • 后记

前言

上一期数据结构专栏我们学习了顺序表后:C语言数据结构顺序表

在运用时,细心的同学可能会发现,如果要头插、尾插或者任意位置。如果原先的空间已经被占满了,你是需要扩容的,动态链表扩容往往是2倍,但是扩容后,如果后面没有使用完全扩容后空间就会造成空间浪费,为了解决这个问题,我们今天将学习链表。

提示:以下是本篇文章正文内容,下面案例可供参考

一、链表的概念及结构

1.概念

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

(图片来自比特就业课)

上图是一个普通链表,它物理上是不连续的,那怎么让这些数据建立联系呢?链表每个位置会存放两个数据——1个是数据,另1个是指针

typedef int SLTDataType;//这里是方便将来改数据类型,在这里把int修改一下即可
typedef struct SlistNode//一个位置放多个数据用结构体
{
	int data;
	struct SlistNode*next;//指针是指向下一个节点,下一个节点也是结构体,所以这里是结构体指针

}SLTNode;//将结构体类型名用SLTNode重命名

我们用一张图来深入了解一下它的逻辑结构:

图示是链表的三个相连元素,第一个元素存放数据1和第二个元素的地址,然后以此类推。那到这里可能有同学会问:“那每个都是有存放下一个地址,最后一个节点存放谁的地址?”是这样的,最后一个节点存放的是NULL空指针,空指针也是作为链表结尾的标记
到这里,我们知道,链表的每个节点放的是当前节点内容和下一个节点的指针,那我们怎么找到这条链表呢?毕竟你第一个节点放的是第二个节点的指针,事实上,我们会定义一个指针指向第一个节点,以此来确定该链表

二、链表的使用

1.遍历整个链表

代码如下(示例):

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

我们以这张图来理解一下这段代码,我们创建一个结构体指针cur,并把链表首元素地址赋给cur,也就是说,cur指向了链表的首节点,我们知道首节点里是有第二个节点的地址的,我们通过cur = cur->next;可以找到第二段节点,以此类推。

2.尾插

(图片来自比特就业课)

想要尾插一个节点,我们首先要malloc(扩容函数)一块空间出来,开辟出那块空间之后,要把新节点与链表连接起来,就需要链表尾部节点的地址,我们用循环遍历得到,然后把新节点地址赋给之前链表的尾部节点即可
代码如下(示例):

void SListPushBack(SLTNode**pphead, SLTDataType x)
{
	SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
	//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
	newnode->data = x;
	newnode->next = NULL;
	if (*pphead == NULL)//防止一开始链表里面什么也没有,
	{//由于pphead是头节点的地址的地址,*解引用一下得到头结点的地址
		*pphead = newnode;
	}
	else
	{
		SLTNode*tail = *pphead;
		while (tail->next != NULL)//寻找尾节点的指针
		{
			tail = tail->next;
		}
		tail->next = newnode;//新增的节点地址赋给尾结点存储
	}
}

这里为什么用二级指针传参?解释如下:
我们进行尾插时,要先找到链表所在嘛,然后这个是靠链表头节点地址确定的,你传了一个地址过去,注意这个地址是实参,你实参过去函数里会再创建一个形参也是这个地址,然后进行操作,改变的是形参里的东西,实参不会受影响。这里也就是传值调用和传址调用的区别,为了形参可以影响到实参,我们用传址调用,也就是传地址的地址

3.头插

(图片来自比特就业课)

我们假设现在要在一个链表前面插入0这个数据(如上图所示),0所在地址为0x0006708,那是不是要把原链表首节点地址放到0这个节点里面,然后头节点地址更新为0这个新节点的地址。如下图:

ps:plist/phead表示链表首节点地址
代码如下(示例):

void SListPushFront(SLTNode**pphead, SLTDataType x)
{
	SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
	//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
	newnode->data = x;//插入节点的数据
	newnode->next = *pphead;//把原先节点首元素地址赋给新节点
	*pphead = newnode;//新节点首元素地址用新插入的节点地址赋值
}

4.头删

关于头删,很多同学可能有一种想法,就是直接把头结点free(对动态分配的内存进行释放)掉,剩下的就是我们需要的链表。但这里会有一个问题,就是你把头节点去了,因为链表是用地址连接的,我们原本是用头节点地址找到该链表,现在头节点去掉了,我们怎么知道剩下的链表在哪里。所以这里的解决方案是,先把第二个节点地址保存起来然后再释放掉头节点
代码如下(示例):

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

5.尾删

尾删和头删有同样的问题,我们能不能直接把尾部节点给free掉?答案也是不可以的,同学你细想一下,尾部节点是不是连接着倒数第二个节点,而倒数第二个节点保存着尾结点地址,你把尾结点free掉了,那倒数第二个节点保存的就是野指针了,这显然是不行的,所以我们需要把倒数第二个节点存储的指针变成空指针,然后free掉尾结点

void SListPopBack(SLTNode**pphead)
{
	if (*pphead == NULL)//如果节点本来就没有,那就没有删除的说法了
	{
		return;
	}
	else if ((*pphead)->next == NULL)//如果本来链表只有一个节点,你删除一个,之前会有一个指针指向首节点来记录这个链表,你现在把唯一的节点删除了,那个记录链表的指针就成野指针了
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SLTNode*prev = NULL;//prev为tail前个节点指针
		SLTNode*tail = *pphead;
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		free(tail);
		prev->next = NULL;
	}
}

这里还是有注意的点的,比如你要删除的链表本来就是空链表,删除就无从谈起了,还有就是原先链表只有一个节点,你删除一个,之前会有一个指针指向首节点来记录这个链表,你现在把唯一的节点删除了,那个记录链表的指针就成野指针了

6.任意位置插入数据

上图红色是原有链表,我们要在2和3直接插入一个30怎么做?首先我们要把2、30、3这3个节点连接起来,也就是说,2节点要指向30这个节点,30这个节点要指向3这个节点。这里如何操作呢?我们需要设计一个函数先找到2节点和3节点的地址,然后进行插入操作
查找函数和插入函数如下

SLTNode* SListFind(SLTNode*phead, SLTDataType x)//查找链表
{
	SLTNode*cur = phead;
	while (cur)//非空指针则进行循环
	{
		if (cur->data == x)//给定链表中一个数据进行查找
		{
			return cur;//返回所查找位置指针
		}
		cur = cur->next;
	}
	return NULL;
}
void SListInsert(SLTNode**pphead,SLTNode*pos, SLTDataType x)//在pos前插入x,x是要插入的数
{
	if (pos == *pphead)//原链表只有1个节点
	{
		SListPushFront(pphead, x);
	}
	else
	{
		SLTNode*newnode = (SLTNode*)malloc(sizeof(SLTNode));
		//开辟一块大小为一个SLTNode类型的空间出来,并强制转换成结构体指针赋给newnode
		newnode->data = x;
		newnode->next = NULL;//开辟1个新节点
		SLTNode*prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = newnode;
		newnode->next = pos;//把新节点连接上去
	}
}

两个函数一起调用是这样的

if (pos)//防止该位置是空指针
	{
		SlistInsert(&plist, pos, 30);
	}
	SListPrint(&plist);
}

7.任意位置删除数据

比如说我们现在要把3删除,那这里就会出现一个需要:3删除了,要把2和4连接起来,然后把3节点释放掉

void SListErase(SLTNode**pphead, SLTNode*pos)//删除pos位置的值
{
	SLTNode*prev = *pphead;
	while (prev->next != pos)//找到3节点的位置
	{
		prev = prev->next;
	}
	prev->next = pos->next;//prev是2,pos是3,pos->next是4,要把4接上2
	free(pos);//2和4接上之后就可以free掉3了
}

ps:上述prev是2,pos是3这种是方便同学你理解,并不特指它真的是2和3

然后这里进行函数调用的时候,依然需要进行find定位一下需要删除位置地址

SLTNode*pos = SListFind(plist, 3);
	if (pos)//防止该位置是空指针
	{
		SListErase(&plist, pos);
	}

后记

链表是数据结构非常重要的一块知识,本文着重介绍了链表的接口函数,下一期笔者会更新特殊链表的使用,点赞三连可以加快更新速度哦~更多关于C语言数据结构单链表接口函数的资料请关注我们其它相关文章!

(0)

相关推荐

  • 利用C语言实现顺序表的实例操作

    本文实例讲述了C语言实现顺序表(线性表)的方法.分享给大家供大家参考,具体如下: 一:顺序表代码实现 #ifndef _SEQ_LIST_H #define _SEQ_LIST_H #include <stdio.h> #include <stdlib.h> #include <assert.h> #include <string.h> #define ElemType float //以float类型测试算法通用性,而不是以惯用的int #define I

  • C语言编程数据结构线性表之顺序表和链表原理分析

    目录 线性表的定义和特点 线性结构的特点 线性表 顺序存储 顺序表的元素类型定义 顺序表的增删查改 初始化顺序表 扩容顺序表 尾插法增加元素 头插法 任意位置删除 任意位置添加 线性表的链式存储 数据域与指针域 初始化链表 尾插法增加链表结点 头插法添加链表结点 打印链表 任意位置的删除 双向链表 测试双向链表(主函数) 初始化双向链表 头插法插入元素 尾插法插入元素 尾删法删除结点 头删法删除结点 doubly-Linked list.c文件 doubly-Linkedlist.h 线性表的定

  • C语言编程数据结构带头双向循环链表全面详解

    目录 前言 一.什么是带头循环双向链表 二.链表初始化 三.链表接口函数 1.尾插 2.头插 3.头删 4.尾删 5.任意位置插入数据 6.任意位置删除数据 四.打印链表 总结 前言 上一篇数据结构专栏:C语言数据结构单链表接口函数全面讲解教程 我们介绍了单链表的各个接口函数,大家可能会发现单链表存在一些缺陷:比如它一个节点要存储数据+下一个节点地址,占用的空间要远多于顺序表:并且由于单链表是无法从后往前找的,如果你想进行尾删这样的操作,你必须从第一个节点往后找,你的时间复杂度一定是O(n).

  • C语言实现的顺序表功能完整实例

    本文实例讲述了C语言实现的顺序表功能.分享给大家供大家参考,具体如下: seqlist.h #ifndef __SEQLIST_H__ #define __SEQLIST_H__ #include<cstdio> #include<malloc.h> #include<assert.h> #define SEQLIST_INIT_SIZE 8 #define INC_SIZE 3 //空间增量的大小 typedef int ElemType; typedef struc

  • 用C语言举例讲解数据结构中的算法复杂度结与顺序表

    数据结构算法复杂度 1.影响算法效率的主要因素 (1)算法采用的策略和方法: (2)问题的输入规模: (3)编译器所产生的代码: (4)计算机执行速度. 2.时间复杂度 // 时间复杂度:2n + 5 long sum1(int n) { long ret = 0; \\1 int* array = (int*)malloc(n * sizeof(int)); \\1 int i = 0; \\1 for(i=0; i<n; i++) \\n { array[i] = i + 1; } for(

  • C语言编程简单却重要的数据结构顺序表全面讲解

    目录 前言 一.线性表定义 二.顺序表实现 1概念及结构 2静态顺序表 2.1实现顺序表接口,第一步要对顺序表进行初始化 2.2对顺序表的增删查改的接口函数(以尾插为例) 3动态顺序表 3.1动态顺序表初始化 3.2动态顺序表-尾插 3.3动态顺序表-头插 3.4动态顺序表-尾删 3.5动态顺序表-头删 3.6动态顺序表-任意位置插入数据 3.7动态顺序表-任意位置删除数据 结束 前言 本文主要介绍顺序表的定义和常见静态顺序表的用法. 一.线性表定义 线性表(line list)是n个具有相同特

  • c语言实现顺序表的基本操作

    数据结构顺序表操作 复制代码 代码如下: #include <stdio.h>#include <stdlib.h>#include <malloc.h>#define LIST_INIT_SIZE 100#define LISINCREMENT 10#define ElemType int#define Status inttypedef struct Sq{ ElemType *elem; int length; int listsize;}SqList;Statu

  • C语言数据结构单链表接口函数全面讲解教程

    目录 前言 一.链表的概念及结构 1.概念 二.链表的使用 1.遍历整个链表 2.尾插 3.头插 4.头删 5.尾删 6.任意位置插入数据 7.任意位置删除数据 后记 前言 上一期数据结构专栏我们学习了顺序表后:C语言数据结构顺序表 在运用时,细心的同学可能会发现,如果要头插.尾插或者任意位置.如果原先的空间已经被占满了,你是需要扩容的,动态链表扩容往往是2倍,但是扩容后,如果后面没有使用完全扩容后空间就会造成空间浪费,为了解决这个问题,我们今天将学习链表. 提示:以下是本篇文章正文内容,下面案

  • C语言数据结构实现链表去重的实例

    C语言数据结构实现链表去重的实例 题目及分析 链表去重 时间限制 300 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 给定一个带整数键值的单链表L,本题要求你编写程序,删除那些键值的绝对值有重复的结点.即对任意键值K,只有键值或其绝对值等于K的第一个结点可以被保留.同时,所有被删除的结点必须被保存在另外一个链表中.例如:另L为21→-15→-15→-7→15,则你必须输出去重后的链表21→-15→-7.以及被删除的链表-15→15. 输入格式: 输入

  • C语言实现单链表实现方法

    C语言实现单链表实现方法 链表和我们之前实现过的顺序表一样,都是简单的数据结构,链表分为单向链表.双向链表.循环链表.而单向链表又分为两种实现方法,一种为带头节点的单链表,一种为不带头节点的单链表.我们来具体看看不带头节点的单链表的实现 单链表:它是一种链式存储的线性表,用一组地址任意的存储单元存放线性表的数据元素,称存储单元为一个节点. 今天我们来实现一些单链表的简单接口 先看看单链表的结构: (为了通用性,我们将类型重命名为DataType) typedef int DataType; //

  • C语言单链表的图文示例讲解

    目录 一.单链表的结构 二.单链表的函数接口 1. 申请结点及打印单链表 2. 尾插尾删 3. 头插头删 4. 中间插入和删除 1. 在 pos 指向的结点之后插入结点 2. 在 pos 指向的结点之前插入结点 3. 删除 pos 指向的结点的后一个结点 4. 删除 pos 指向的结点 6. 查找 7. 销毁单链表 在上一篇所讲述的 动态顺序表 中存在一些缺陷 1.当空间不够时需要扩容,扩容是有一定的消耗的 如果每次空间扩大一点,可能会造成空间的浪费,而空间扩小了,又会造成频繁的扩容2.在顺序表

  • 浅谈PHP链表数据结构(单链表)

    链表:是一个有序的列表,但是它在内存中是分散存储的,使用链表可以解决类似约瑟夫问题,排序问题,搜索问题,广义表 单向链表,双向链表,环形链表 PHP的底层是C,当一个程序运行时,内存分成五个区(堆区,栈区,全局区,常量区,代码区) 规定:基本数据类型,一般放在栈区 复合数据类型,比如对象,放在堆区 定义一个类Hero 定义成员属性排名 $no 定义成员属性姓名 $name 定义成员属性昵称 $nickname 定义成员属性 $next,是一个引用,指向下一个Hero对象 定义构造函数,传递参数:

  • C语言实现单链表反转

    一.理解指针 看懂链表的结构并不是很难,但是一旦把它和指针混在一起,就很容易让人摸不着头脑.所以,要想写对链表代码,首先就要理解好指针. 有些语言有"指针"的概念,比如 C 语言:有些语言没有指针,取而代之的是"引用",比如 Java.Python.不管是"指针"还是"引用",实际上,它们的意思都是一样的,都是存储所指对象的内存地址. 将某个变量赋值给指针,实际上就是将这个变量的地址赋值给指针,或者反过来说,指针中存储了这个变

  • C语言使用单链表实现学生信息管理系统

    本文实例为大家分享了C语言使用单链表实现学生信息管理系统,供大家参考,具体内容如下 初学数据结构,记录一下学习过程. 运行结果如图: 1.运行界面 2.录入学生信息 3.按照总分进行排序 代码如下: #define ERROR 0 #define OK 1 #define OVERFLOW -1; typedef int ElemType; typedef int Status; #include<stdio.h> #include<stdlib.h> #include<ma

  • C语言实现单链表的基本功能详解

    1.首先简单了解一下链表的概念: 要注意的是链表是一个结构体实现的一种线性表,它只能从前往后,不可以从后往前(因为next只保存下一个节点的地址).在实现单链表的操作时,需要用指针来操作.很简单,注释写的很详细,欢迎大家指正哈哈哈哈~之前写的太烂了重新写了一下..... 2.代码展示: #include <stdio.h> #include <assert.h> #include <stdlib.h> typedef struct linklist { int data

  • ​​​​​​​C语言实现单链表基本操作方法

    目录 存储结构 基本功能 头插法创建单链表 尾插法创建单链表 获取指定位置的元素 在指定位置插入元素 删除指定位置的元素 获取单链表的长度 合并两个非递减的单链表 晴链表 遍历打印单链表 附上完整代码 存储结构 typedef int dataType://爱护据类型 typedef struct Node { DataType data; // 结点数据 struct Node *next; // 指向下一个结点的指针 } Node, *LinkList; 基本功能 头插法创建单链表void

  • java 数据结构单链表的实现

    java 数据结构单链表的实现 单链表实现链表的打印及元素删除操作,链表的实现主要是next属性的定义,将一堆节点关联起来的.实现简单的链表如下: public class LinkNode { private int value; private LinkNode next; public LinkNode(int x) { value = x; } public LinkNode getNext(){ return next; } public void setNext(LinkNode n

随机推荐