【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表)

这篇具有很好参考价值的文章主要介绍了【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表

💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤
📃个人主页 :阿然成长日记 👈点击可跳转
📆 个人专栏: 🔹数据结构与算法🔹C语言进阶
🚩 不能则学,不知则问,耻于问人,决无长进
🍭 🍯 🍎 🍏 🍊 🍋 🍒 🍇 🍉 🍓 🍑 🍈 🍌 🍐 🍍


前言

🎸小伙伴们,又见面了🌻 🌺 🍁 🍃 前面我们学习啦顺序表,其实顺序表的时间复杂度是很高的,尤其是在插入,删除等问题上,需要移动整个数组,十分麻烦费时。有没有更好的办法呢????当然有呀,就是链表,也是本篇博客要详细讲解的。

一、链表的概念

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

链表就像是一列火车,链表中的每一个节点,就像是火车的一节节车厢。
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
图1.1
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表图1.2

上面两幅图片生动地解释了链表的物理结构。想必看到这里已经对链表有了初步的认识。

二、特点

1️⃣ 链式结构在逻辑上是连续的,但在物理层上不一定连续。
2️⃣节点一般都是从堆上申请出来的一块空间。
3️⃣从堆上申请的空间,按照它的规则来进行分配,两次申请的空间,不一定连续。

三、链表的分类

1.单向或者双向
2. 带头或者不带头
3. 循环或者非循环

四、单向链表的结构体

❌误区:以下这种结构体定义会报错,那么是为什么呢?

typedef int SLTDataType;
typedef struct SListNode
{
	SLTDataType data;
	SLTNode* next;//错误

}SLTNode;

我们的typedef关键字给结构体重新命名为SLTNode,但是他是在结构体最后才生效,如果现在就在结构体中使用新命名,那么就会找不到。

👍正解是:

typedef int SLTDataType;
typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;

 }SLTNode;

1.node:是存储的数据;
2.next 的类型是一个节点型的指针变量,它保存的是下一个节点的地址,即指向下一个节点

命名规范:

当我们在给结构体命名或者是函数的命名我们都应该使用用英文或者英文的简写来进行命名这样有利于人们的理解。例如单链表英文名:single List table,所以我给节点命名为SLTNode.

二级指针

在下面的学习中,会使用二级指针,不太清楚的小伙伴,可以去看我的📋C进阶专栏中的👉高级指针一篇

❗️注意事项

我们现在定义的头指针在函数结束之后都会销毁,因为它存在栈上。我们的每一个节点是使用动态内存函数在堆上进行开辟如果不进行free释放那么它会持续保存到程序结束。

五、函数实现

1.单链表的打印

//打印单链表
void PrintSlistTable(SLTNode* phead)
{
	SLTNode* cur = phead;
	while (cur != NULL)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL");
}

2.单链表的头插

头插思路分析:
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表

头插代码

//头插
void SLTPusFront(SLTNode** pphead, SLTDataType x)//放入新插入节点
{
	SLTNode* newnode = CreatNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

这里有很多小伙伴都不知道为什么使用了二级指针。因为在传参时我们使用的是结构体地址传参,这样能节省空间,提高效率,传入的是一级指针phead的地址,所以我们需要使用二级指针pphead来接收。

3.单链表的尾插

尾插思路分析:

【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
尾插代码:

//尾插
void SLTPusBack(SLTNode**  pphead, SLTDataType x)
{
	SLTNode* newnode = CreatNode(x);
	if (*pphead == NULL)
	{
		//改变的结构体的指针,所以要用二级指针 
		*pphead = newnode;
	}
	else
	{
		SLTNode* tail = *pphead;
		while (tail->next != NULL)
		{
		tail = tail->next;
		}
		//改变结构体,用结构体指针即可 
	tail->next = newnode;
	}
}

4.单链表的头删

思路:
一个节点和多个节点处理方式相同
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

//头删
void PopFront(SLTNode** pphead)
{
	assert(*pphead);
	SLTNode* cur = (*pphead)->next;
	free(*pphead);
	*pphead = cur;
}

1️⃣ 定义一个cur临时指针用来指向头节点的下一个节点.SLTNode* cur = (*pphead)->next;
2️⃣ 释放 *pphead即(删除第一个节点)free(*pphead);
3️⃣ 在将 *pphead指向第二节点*pphead = cur;

5.单链表尾删

思路:
1.如果没有节点,则直接释放头指针所指向的内容
2.
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

//尾插
void SLTPusBack(SLTNode**  pphead, SLTDataType x)
{
	SLTNode* newnode = CreatNode(x);
	if (*pphead == NULL)
	{
		//改变的结构体的指针,所以要用二级指针 
		*pphead = newnode;
	}
	else
	{
		SLTNode* tail = *pphead;
		while (tail->next != NULL)
		{
		tail = tail->next;
		}
		//改变结构体,用结构体指针即可 
	tail->next = newnode;
	}
}

6.在pos位置之前插入x

思路:
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

//在pos位置之前插入x
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(*pphead);
	assert(pos);

	if (*pphead == pos)
	{
		//头插;
		SLTPusFront(pphead, x);
	}
	else
	{
		//定义一个临时指针cur指向头指针,为了从头开始遍历各个节点找pos,而不会改变头指针pphead的指向位置。
		SLTNode* cur = *pphead;
		while (cur->next != pos)
		{
			cur = cur->next;
		}
		SLTNode* newNode = CreatNode(x);
		newNode->next = cur->next;
		cur->next = newNode;
		free(cur);
		cur = NULL;
	}
}

定义一个临时指针cur指向头指针,用来从头开始遍历各个节点找pos,
头指针pphead的指向位置不能变,不然就找不到头了。

7.在pos位置之后插入x

思路:

【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

在pos位置之后插入x
void SLTInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{
	assert(*pphead);
	SLTNode* newNode = CreatNode(x);
	newNode->next = pos->data;
	pos->next = newNode;
}

8.删除pos位置 节点

思路:
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

//删除POS位置 
void SLTErase(SLTNode** pphead, SLTNode* pos)
{
	assert(*pphead);
	assert(pos);

	if (*pphead == pos)
	{
		//头删
		SLTPopFront(pphead);
	}
	else
	{
		SLTNode* cur = *pphead;
		while (cur->next == pos)
		{
			cur = cur->next;

		}
		pos->next = cur->next;
		free(pos);
	}
}

9.删除pos位置之后的节点

思路:
【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表),数据结构与算法,数据结构,链表
代码:

//删除POS之后的位置 
void SLTEraseAfter(SLTNode** pphead, SLTNode* pos)
{

		assert(pphead);
		assert(pos);
		assert(pos->next);

		SLTNode* cur = pos->next->next;
		free(pos->next);
		pos->next = cur;

}

10.单链表的查找

代码:文章来源地址https://www.toymoban.com/news/detail-628750.html

//单链表的查找 

SLTNode*  SLTSrech(SLTNode** pphead, SLTDataType x)
{
	SLTNode* cur = *pphead;
	while (cur->next!= NULL)
	{
		if (cur->data == x)
			return cur;
		cur = cur->next;
	}
	return cur;
}

到了这里,关于【数据结构】之十分好用的“链表”赶紧学起来!(第一部分单向链表)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据结构——线性数据结构(数组,链表,栈,队列)

    数组(Array) 是一种很常见的数据结构。它由相同类型的元素(element)组成,并且是使用一块连续的内存来存储。 我们直接可以利用元素的索引(index)可以计算出该元素对应的存储地址。 数组的特点是: 提供随机访问 并且容量有限。 2.1. 链表简介 链表(LinkedList) 虽然是

    2024年02月11日
    浏览(33)
  • 【数据结构】详解链表结构

    上篇博客已经介绍了顺序表的实现:【数据结构】详解顺序表。最后在里面也谈及了顺序表结构的缺陷,即 效率低,空间浪费 等等问题,那么为了解决这些问题,于是乎我们引入了链表的概念,下面将对链表结构进行讲解 首先肯定会问,到底什么是链表? 链表的概念 : 链

    2024年02月05日
    浏览(33)
  • 【数据结构】链表的回文结构

    单链表的操作算法是笔试面试中较为常见的题目。 本文将着重介绍平时面试中常见的关于链表的应用题目,马上要进行秋招了。希望对你们有帮助 _ 😀 对于一个链表,请设计一个时间复杂度为O(n),额外空间复杂度为O(1)的算法,判断其是否为回文结构。 给定一个链表的头指针

    2024年02月12日
    浏览(39)
  • 【数据结构】浅谈数据结构-链表【思路+例题学习】

    🏆今日学习目标: 🍀学习算法-数据结构-链表 ✅创作者:贤鱼 ⏰预计时间:30分钟 🎉个人主页:贤鱼的个人主页 🔥专栏系列:算法 🍁贤鱼的个人社区,欢迎你的加入 贤鱼摆烂团 链表是一种物理存储单元上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中

    2024年01月21日
    浏览(39)
  • 【数据结构】每天五分钟,快速入门数据结构(二)——链表

    目录 一 构建一个单向链表 二 特点 三 时间复杂度 四 相关算法 1.判断链表是否成环及成环位置 2.链表反转 五 Java中的LinkedList 类 1.使用 2.LinkedList 方法 长度无限 适合任意位置插入和删除频繁的场景 物理上可以不连续 访问、插入、删除 的时间复杂度均为O(n) 在头部插入元素

    2024年02月21日
    浏览(31)
  • 【数据结构】反转链表、链表的中间节点、链表的回文结构(单链表OJ题)

    正如标题所说,本文会图文详细解析三道单链表OJ题,分别为:  反转链表 (简单)  链表的中间节点 (简单)  链表的回文结构 (较难) 把他们放在一起讲的原因是:  反转链表 和  链表的中间节点 是  链表的回文结构 的基础 为什么这样说?请往下看: 目录 1. 反转链

    2024年02月13日
    浏览(39)
  • 数据结构:详解【链表】的实现(单向链表+双向链表)

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

    2024年03月26日
    浏览(53)
  • Python数据结构与算法-数据结构(列表、栈、队列、链表)

    数据结构是指相互之间存在这一种或者多种关系的数据元素的集合和该集合中元素之间的关系组成。 简单来说,数据结构就是设计数据以何种方式组织并存储在计算机中。 比如:列表、集合与字典等都是一种数据结构。 N.Wirth:“程序=数据结构+算法” 数据结构按照其 逻辑结

    2024年02月08日
    浏览(36)
  • 【数据结构】链表的分类和双向链表

    本篇是基于上篇单链表所作,推荐与上篇配合阅读,效果更加 http://t.csdnimg.cn/UhXEj 链表的结构非常多样,以下情况组合起来就有8种(2 x 2 x 2)链表结构: 我们一般叫这个头为哨兵位 我们上回讲的单链表就是不带头单项不循环链表。 今天我们要讲带头双向循环的链表。 不过

    2024年01月25日
    浏览(27)
  • 数据结构入门 — 链表详解_双向链表

    数据结构入门 — 双向链表详解 博客主页链接:https://blog.csdn.net/m0_74014525 关注博主,后期持续更新系列文章 文章末尾有源码 *****感谢观看,希望对你有所帮助***** 第一篇:数据结构入门 — 链表详解_单链表 第二篇:数据结构入门 — 链表详解_双向链表 第三篇:数据结构入门

    2024年02月11日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包