【数据结构】- 链表之单链表(上)

news2024/11/17 13:33:05

文章目录

  • 前言
  • 一、链表
    • 1.1链表的概念及结构
    • 1.2链表的分类
  • 二、单链表(上)
    • 2.1单链表的实现
    • 2.2单链表实现的两种结构解析
    • 2.3单链表的接口实现
      • 2.3.1头插
      • 2.3.2温馨提醒 宝子~
      • 2.3.3头插完整版代码
      • 2.3.4尾插
      • 2.3.5温馨提醒 宝子~
      • 2.3.6总而言之
  • 总结


前言

“偶尔失意 是为了压住翘起的尾巴”
本章是关于数据结构中的链表之单链表(上)


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

一、链表

1.1链表的概念及结构

概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的
在这里插入图片描述

1.2链表的分类

实际中链表的结构非常多样,以下情况组合起来就有8种链表结构:

  • 单向或者双向
    在这里插入图片描述

  • 带头或者不带头
    在这里插入图片描述

  • 循环或者非循环
    在这里插入图片描述

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:
在这里插入图片描述

  1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。
  2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了,后面我们代码实现了就知道了

二、单链表(上)

2.1单链表的实现

//SList.h
#include<stdio.h>

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

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

2.2单链表实现的两种结构解析

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

2.3单链表的接口实现

2.3.1头插

void SLPushFornt(SLTNode* phead, SLTDataType x)
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}
	newnode->data = x;
	newnode->next = NULL;
	newnode->next  = phead;
	phead = newnode;
}

逻辑分析:
在这里插入图片描述
物理分析:
在这里插入图片描述
注意:
上述代码中newnode是一个局部变量,phead是一个形参出了作用域就会被销毁所以我们需要在外面留一个指针来找到这个节点的地址

2.3.2温馨提醒 宝子~

当我们在外面留下一个指针来找这个节点并打印的时候我们发现打印出来是个NULL,这是怎么回事呢?
在这里插入图片描述
在这里插入图片描述
举个例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
void Swap(int* p1, int* p2)
{
	int tmp = *p1;
	*p1 = *p2;
	*p2 = tmp;
}

void Swap1(int** pp1, int** pp2)
{
	int* tmp = *pp1;
	*pp1 = *pp2;
	*pp2 = tmp;
}

int main()
{
	int a = 0;
	int b = 1;
	Swap(&a, &b);
	printf("a=%d,b=%d\n", a, b);
	int c = 2;
	int d = 8;
	int* px = &c;
	int* py = &d;
	printf("px=%p,py=%p\n", px, py);
	Swap1(&px, &py);
	printf("px=%p,py=%p\n", px, py);
	return 0;
}

在这里插入图片描述

2.3.3头插完整版代码

分为三个模块分别是SList.h SList.c Test.c

//SList.h
#include<stdio.h>
#include<stdlib.h>

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

void SLTPrint(SLTNode* phead);
void SLPushFornt(SLTNode** pphead, SLTDataType x);




//SList.c
#include"SList.h"

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

SLTNode* BuyTNode(SLTDataType x)//因为头插尾插等都需要maolloc一块新空间所以我们分装成一个函数
{
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
		return;
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

void SLPushFornt(SLTNode** pphead, SLTDataType x)
{
	SLTNode* newnode = BuyTNode(x);
	newnode->next  = *pphead;
	*pphead = newnode;
}




//Test.c
#include"SList.h"

void TestSList()
{
	SLTNode* plist = NULL;
	SLPushFornt(&plist, 1);
	SLPushFornt(&plist, 2);
	SLPushFornt(&plist, 3);
	SLPushFornt(&plist, 4);

	SLTPrint(plist);
}

int main()
{
	TestSList();
	return 0;
}

2.3.4尾插

void SLPushBack(SLTNode* phead, SLTDataType x)
{
	SLTNode* tail = phead;
	while (tail->next != NULL)
	{
		tail = tail->next;
	}
	SLTNode* newnode = BuyTNode(x);
	tail->next = newnode;
}

在这里插入图片描述

2.3.5温馨提醒 宝子~

(1).下面这段代码存在两个问题:

  • 内存泄露
  • 链表并没有链接起来
void SLPushBack(SLTNode* phead, SLTDataType x)
{
	SLTNode* tail = phead;
	while (tail->next != NULL)
	{
		tail = tail->next;
	}
	SLTNode* newnode = BuyTNode(x);
	tail->next = newnode;
}

在这里插入图片描述

(2).下面这段代码存在问题:
我们通过运行代码可以知道,尾插不用二级指针也是可以的
在这里插入图片描述
但是我们需要考虑两个问题:

  • 尾插时前面是非空链表:

在这里插入图片描述

  • 尾插时前面是空链表:
    (1). 使用二级指针在这里插入图片描述
    (2). 使用一级指针
    在这里插入图片描述
  • 总结图:所以总而言之为了满足两者的条件我们使用二级指针
  • 其实就类似于头插,我们头插时并不是开始就有数字开始也是NULL,所以使用二级指针(传址) * pphead指向plist,然后开辟空间newnode赋给*pphead,然后链接,第一次头插完成后, * pphead和newnode销毁,但是plist已经通过传地址改变了里面的地址指向了第一个节点空间地址,当第二次插入数字时,把plist传址给 * pphead;然后开辟了一块新空间newnode,注意哦plist经过上次已经指向了第一个节点而把plist传址给 *pphead这时 *pphead就指向第一个节点的地址只需要让 *pphead赋给头插的那个newnode->next就链接起来了然后 *pphead = newnode,这样出了作用域都销毁而plist也就指向了头插的那个节点的位置;
  • 但是不使用二级指针,只是把plist中的地址赋给phead,phead改变并不影响plist因为出了作用域就会销毁
    在这里插入图片描述

2.3.6总而言之

  • 要改变int,传int的地址
  • 要改变int * ,传int * 的地址
  • 要改变结构体,传结构体的地址
  • 要改变结构体指针,传结构体的指针的地址

总结

Ending,今天的链表之单链表(上)的内容就到此结束啦~,如果后续想了解更多,就请关注我吧,一键三连哦 ~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/451992.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

web事件循环

事件循环的应用&#xff1a;计时器 promise ajax node 单线程是异步产生的原因&#xff0c;事件循环时异步的实现方式 1.浏览器进程模型 进程&#xff1a;程序运行需要自己专属的内存空间&#xff0c;可以把这块内存空间简单的理解为进程。 每个应用至少又一个进程&#xff…

头文件stdafx.h的作用(常见错误),以及如何在空项目中添加该头文件

头文件stdafx.h的作用&#xff08;常见错误&#xff09;&#xff0c;以及如何在空项目中添加该头文件 预编译头文件 stdafx.hfatal error C1083: 无法打开包括文件:“stdafx.h”: No such file or directoryfatal error C1083: 无法打开预编译头文件:“Debug\Win32RegistryClas…

【LeetCode】72. 编辑距离

72. 编辑距离&#xff08;困难&#xff09; 思路 状态定义&#xff1a;「dp[i][j] 表示第一个字符串到 i &#xff0c;第二个字符串到 j&#xff0c;要想使得 word1 word2 &#xff0c;最少的修改次数」。状态转移方程&#xff1a; 当第 i 位和第 j 位对应的字符相同时&#x…

【youcans 的 OpenCV 学习课】21. Haar 小波变换与 Haar 特征检测(上)

专栏地址&#xff1a;『youcans 的图像处理学习课』 文章目录&#xff1a;『youcans 的图像处理学习课 - 总目录』 【youcans 的 OpenCV 学习课】21. Haar 小波变换与 Haar 特征检测&#xff08;上&#xff09; 1. 小波变换1.1 小波变换基本概念例程 17_1&#xff1a;常用小波族…

学会这几个Word技巧,让你办公省时又省力(一)

如果在学习&#xff0c;或者工作上&#xff0c;你经常需要用到Word文档&#xff0c;那下面这几个Word技巧一定要学习收藏&#xff0c;提升工作效率不是一点点。 ​1. 快速定位对象 有时候文档的内容很长&#xff0c;涉及很多表格、图片、图表&#xff0c;想要检查一下所有的图…

linux从入门到精通 第一章centos7里tomcat,jdk,httpd,mysql57,mysql80的安装

配置centos运行环境 一 安装httpd,tomcat,jdk,mysql1 安装httpd2 安装tomcat3 安装jdk 三 MySql的安装1 克隆出来两台虚拟机2 配置虚拟机3 链接xhsell4 链接xftp5 mysql8的安装6 mysql5.7的安装 一 安装httpd,tomcat,jdk,mysql 1 安装httpd 下载httpd yum -y install httpd关…

微信小程序php+vue校园达达互助平台快递代取系统

校园快递互助平台所要实现的功能分析&#xff0c;对于现在网络方便&#xff0c;校园快递互助平台要实现管理员、学生、兼职者可以直接在平台上进行查看自己所需数据信息&#xff0c;这样既能节省管理的时间&#xff0c;不用再像传统的方式&#xff0c;如果用户想要进行交流信息…

精读《利用 GPT 解读 PDF》

hatPDF 最近比较火&#xff0c;上传 PDF 文件后&#xff0c;即可通过问答的方式让他帮你总结内容&#xff0c;比如让它帮你概括核心观点、询问问题&#xff0c;或者做观点判断。 背后用到了几个比较时髦的技术&#xff0c;还好有 ChatGPT for YOUR OWN PDF files with LangCha…

【Java实战篇】Day11.在线教育网课平台--RBAC

文章目录 一、用户授权1、RBAC2、资源服务授权流程3、授权相关的数据模型4、查询用户权限5、细粒度授权 二、找回密码与注册1、找回密码2、注册 三、需求&#xff1a;学生选课1、添加选课需求分析2、数据模型设计2、查询课程信息接口3、添加选课接口4、完善controller 一、用户…

每日一个小技巧:1招教你手机消除笔怎么用

在日常生活中&#xff0c;我们经常需要在手机上进行编辑和涂改&#xff0c;但是由于各种原因&#xff0c;我们可能会做出错误或者不满意的修改。这时候&#xff0c;消除笔就派上用场了。消除笔可以帮助我们在不影响其他内容的前提下&#xff0c;对错误或者不满意的修改进行撤销…

java实现大气无风环境污染物扩散模拟

一、扩散公式整理 二、编写java代码实现 String strJson InterpolationUtils.calGaussPlumePoints0(z,height,q,lon,lat, size,scale,airStable); return strJson.replaceAll("NaN","0").replaceAll("Infinity",String.valueOf(q)); String st…

【机器学习】P23 决策树、熵和信息增益

决策树、熵与信息增益 决策树熵信息增益Python 与 决策树 决策树 决策树&#xff08;Decision Tree&#xff09; 是一种基于树形结构的分类算法&#xff0c;它通过一系列的询问&#xff08;也称为测试或判定条件&#xff09;来判断一个数据实例属于哪个类别。 以一个案例贯穿…

Go | 一分钟掌握Go | 2 - 集成开发工具

作者&#xff1a;Mars酱 声明&#xff1a;本文章由Mars酱编写&#xff0c;部分内容来源于网络&#xff0c;如有疑问请联系本人。 转载&#xff1a;欢迎转载&#xff0c;转载前先请联系我&#xff01; 工具介绍 编码是一门传统手艺活&#xff0c;手艺好不好很重要&#xff0c;器…

高精度气象模拟软件WRF实践技术

【原文链接】&#xff1a;高精度气象模拟软件WRF(Weather Research Forecasting)实践技术及案例应用https://mp.weixin.qq.com/s?__bizMzU5NTkyMzcxNw&mid2247538149&idx3&sn3890c3b29f34bcb07678a9dd4b9947b2&chksmfe68938fc91f1a99bbced2113b09cad822711e7f…

开发者必读!常用的二维码生成器 API 推荐

引言 二维码是一种能够存储信息的图形码&#xff0c;它在现代社会中扮演着越来越重要的角色。生成二维码的过程通常需要使用二维码生成器&#xff0c;而现在有很多二维码生成器 API 可以供开发者使用。 在本文中&#xff0c;我们将讨论二维码生成器 API 的工作原理、应用场景…

CHAPTER 6: 《DESIGN A KEY-VALUE STORE》 第6章 《设计一个键值存储》

CHAPTER 6: DESIGN A KEY-VALUE STORE 键值存储(也称为键值数据库)是一种非关系数据库。每一个唯一标识符存储为与其关联值的键。这种数据配对称为“键-值”对。 在一个键-值对中&#xff0c;键必须是唯一的&#xff0c;与该键相关联的值可以是通过密钥访问。键可以是纯文本或…

编译原理个人作业--第五章——基于 编译原理 国防工业出版社 第三版

1 文法 G 1 G_1 G1​为 E → E T ∣ T T → T ∗ F ∣ F F → ( E ) ∣ i E\rightarrow ET|T\\ T\rightarrow T*F|F\\ F\rightarrow(E)|i E→ET∣TT→T∗F∣FF→(E)∣i 请证明 E T ∗ F ET*F ET∗F是他的一个句型(课本写的是ET*T感觉是印错了)&#xff0c;指出它的所有短语…

这份最新阿里、腾讯、华为、字节等大厂的薪资和职级对比,你看过没?

互联网大厂新入职员工各职级薪资对应表(技术线)~ 最新阿里、腾讯、华为、字节跳动等大厂的薪资和职级对比 上面的表格不排除有很极端的收入情况&#xff0c;但至少能囊括一部分同职级的收入。这个表是“技术线”新入职员工的职级和薪资情况&#xff0c;非技术线(如产品、运营、…

pbootcms自动配图出图插件

pbootcms文章无图自动出图配图插件的优点 1、提高文章的可读性和吸引力&#xff1a;插入图片可以丰富文章的内容和形式&#xff0c;增强读者的阅读体验和吸引力&#xff0c;提高文章的点击率和转化率。 2、节省时间和精力&#xff1a;手动添加图片需要花费大量时间和精力去寻找…

【LeetCode: 1416. 恢复数组 | 暴力递归=>记忆化搜索=>动态规划 】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…