当前位置: 首页 > news >正文

【数据结构初阶】单链表SLlist

描述

不同于顺序表,顺序表的数据是存储在一个连续的空间里的

链表它是链接起来的结构体地址。

所以我们不用像顺序表一样先创建一块空间出来,而是创建一个能存数据节点和节点与下一个节点之间的连接

所以:“一个能存数据节点”我们用int Data表示;

与下一个节点之间的连接”:我们用指针连接起来。

组织

打印

为了方便测试,我们先写个打印单链表的函数;

1.这个打印函数需要断言吗?
不需要,即使结构体为空,也能打印,只不过是没有数据而已,这时打印出来的就是空的。如果能打印出来,但是却断言报错,不太合适。

2.怎么打印?
用一个指针cur指向结构体,用while循环打印出来,当cur指向的结构体为空时,停止打印。

3.while的判断条件可以是while(cur->next)吗?
不可以,因为这样最后一个的数据就打印不出来了。

4.在while循环中,让cur指向下一个结构体,可以用cur++吗?
不可以,不同于顺序表,顺序表的数据是存储在一个连续的空间里的,链表它是链接起来的结构体地址,是节点与节点之间的连接,cur++无法指向下一个节点。
 

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

1.尾插

每在插入一个数据之前,首先得为这个结构体开辟一个结点

用malloc开辟,由于插入数据时我们都要进行开辟一个结点的操作,所以我们可以打包成一个函数


进行尾插首先就是要找到尾节点

找尾分两种情况:

1. 当*pphead本身为空时,就直接将newnode插入就可以了;

2. *pphead本身不为空时,只要找到tail->next为空的,那个就是结构体的尾了


当pphead不为空时,找尾while循环的判断条件可以写成这样tail!=NULL与插入结点时tail=newnode吗?
不可以,因为这样就无法保持链接状态

尾插的本质是:原为节点中存储新的尾节点的地址

SLTNode* SLTNewnode(SLTDataType x)
{SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));if (newnode == NULL){perror("melloc fail");return NULL;}newnode->Data = x;newnode->next = NULL;return newnode;}void SLTPushBack(SLTNode** pphead, SLTDataType x)
{//创建节点SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}//情况一:pphead为空if (*pphead == NULL){*pphead = newnode;}//情况二:pphead不为空else{//找到尾节点SLTNode* tail = *pphead;while (tail->next){tail = tail->next;}tail->next = newnode;}
}

2.尾删

1.尾删需要断言吗?
需要,因为如果链表为空,是删不了的;


2.尾删的思路
尾删分三种讨论:

1.如果链表为空,删不了,我们这里断言判断一下
2.链表中只有一个数据
3.链表中的数据为一个以上

void SLTPopBack(SLTNode** pphead)
{assert(*pphead);//只有一个节点if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{//找尾置空SLTNode* tail = *pphead;while (tail->next->next){tail = tail->next;}free(tail->next);tail->next = NULL;}}

3.头插

1.头插需要断言吗?
但是当链表为空的时候,可以插入数据,*pphead是不需要断言的。


2.头插的思路
首先先要创建一个结点,将结点的next与链表的第一个指针链接起来。最后要注意把链表的头给改成newnode。

void SLTPushFront(SLTNode** pphead, SLTDataType x)
{SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}newnode->next = *pphead;*pphead = newnode;
}


4.头删

1.头删需要断言吗?
空链表不能删除,所以*pphead也需要断言。

头删的思路:

void SLTPopFront(SLTNode** pphead)
{assert(*pphead);SLTNode* head = *pphead;*pphead = head->next;free(head);head = NULL;}


5.查找

1.查找需要断言吗?
不需要,链表为空就返回NULL;


2.查找的思路
当链表的cur不为空,就继续逐一比对,找到了就返回cur,没找到就指向下一个,直到cur指向空;

如果还没找到,就返回NULL;

这里的phead用一级指针就可以了,因为不用修改里面的任何值;

SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{SLTNode* cur = phead;while (cur){if (cur->Data == x){return cur;}cur = cur->next;}//如果没找到就返回NULLreturn NULL;
}



6.指定位置后插入

1.需要断言吗?
指定的位置pos不能为空,所以需要断言;


2.思路
创建一个新结点newnode,然后先让newnode->next = pos->next;让newnode的后面链接起来,在让pos和newnode链接起来pos->next = newnode;;

反过来写的话,由于pos->next已经被改了,所以不能是pos与newnode链接,插入就会失败;


void SLTInsertAfter( SLTNode* pos, SLTDataType x)
{assert(pos);SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}newnode->next = pos->next;pos->next = newnode;}



7.指定位置后删除

1.需要断言吗?
指定的位置pos不能为空,所以需要断言;

void SListEraseAfter( SLTNode* pos)
{assert(pos);if (pos->next == NULL){printf("已是最后一个,不能删\n");return;}SLTNode* cur = pos->next;pos->next = pos->next->next;free(cur);cur= NULL;
}

8.链表的销毁

思路
结点逐一free,最后记得把*pphead改为最后的cur。

void SLTDel(SLTNode** pphead)
{assert(*pphead);SLTNode* cur = *pphead;SLTNode* prev = cur;while (cur){prev = cur->next;free(cur);cur = prev;}*pphead = cur;
}

关于其他的一些细节问题

为什么不像顺序表一样写初始化函数?

可写可不写,这里结构体里面的数据比较少,我们在测试代码的时候直接用指针指向了一块空间。

为什么要传二级指针?

想要改变变量的数值而不会因为栈帧的销毁而销毁,就得取到该变量的地址;

什么时候要传二级指针,什么时候要传一级指针?

想要改变变量里的数值就要传址调用,二级指针用来接收一级指针;

如果只是简单的访问就用传值调用

整个程序

.h文件

#define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h>
#include<stdlib.h>
#include<assert.h>typedef int SLTDataType;typedef struct SLTlist
{SLTDataType Data;	struct SLTlist * next;}SLTNode;void SLTPrint(SLTNode* phead);
void SLTPushBack(SLTNode** pphead, SLTDataType x);
void SLTPopBack(SLTNode** pphead);
void SLTPushFront(SLTNode** pphead, SLTDataType x);
void SLTPopFront(SLTNode** pphead);
SLTNode* SListFind(SLTNode* phead, SLTDataType x);
void SLTInsertAfter( SLTNode* pos, SLTDataType x);
void SListEraseAfter( SLTNode* pos);
void SLTDel(SLTNode** pphead);

.c文件

#include"SLlist.h"void SLTPrint(SLTNode* phead)
{SLTNode* cur = phead;while (cur){printf("%d->", cur->Data);cur = cur->next;}printf("NULL\n");
}SLTNode* SLTNewnode(SLTDataType x)
{SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));if (newnode == NULL){perror("melloc fail");return NULL;}newnode->Data = x;newnode->next = NULL;return newnode;}void SLTPushBack(SLTNode** pphead, SLTDataType x)
{//创建节点SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}//情况一:pphead为空if (*pphead == NULL){*pphead = newnode;}//情况二:pphead不为空else{//找到尾节点SLTNode* tail = *pphead;while (tail->next){tail = tail->next;}tail->next = newnode;}
}void SLTPopBack(SLTNode** pphead)
{assert(*pphead);//只有一个节点if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{//找尾置空SLTNode* tail = *pphead;while (tail->next->next){tail = tail->next;}free(tail->next);tail->next = NULL;}}void SLTPushFront(SLTNode** pphead, SLTDataType x)
{SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}newnode->next = *pphead;*pphead = newnode;
}void SLTPopFront(SLTNode** pphead)
{assert(*pphead);SLTNode* head = *pphead;*pphead = head->next;free(head);head = NULL;}SLTNode* SListFind(SLTNode* phead, SLTDataType x)
{SLTNode* cur = phead;while (cur){if (cur->Data == x){return cur;}cur = cur->next;}//如果没找到就返回NULLreturn NULL;
}void SLTInsertAfter( SLTNode* pos, SLTDataType x)
{assert(pos);SLTNode* newnode = SLTNewnode(x);if (newnode == NULL){return;}newnode->next = pos->next;pos->next = newnode;}void SListEraseAfter( SLTNode* pos)
{assert(pos);if (pos->next == NULL){printf("已是最后一个,不能删\n");return;}SLTNode* cur = pos->next;pos->next = pos->next->next;free(cur);cur= NULL;
}void SLTDel(SLTNode** pphead)
{assert(*pphead);SLTNode* cur = *pphead;SLTNode* prev = cur;while (cur){prev = cur->next;free(cur);cur = prev;}*pphead = cur;
}

相关文章:

  • 3 redis实现一个消息中间件
  • “我“摸爬滚打5年,干了测试工程师,现在测试怎么样了...
  • Spark数据倾斜_产生原因及定位处理办法_生产环境
  • 多视图聚类的论文阅读(一)
  • “绵柔的,好喝的”海之蓝畅销20年的经典秘诀:做大众喜爱的好酒
  • Ubuntu 22.04 LTS ffmpeg mp4 gif 添加图片水印
  • [uni-app] uni.showToast 一闪而过问题/设定时间无效/1秒即逝
  • 将 ONLYOFFICE 文档编辑器与 Node.js 应用集成
  • requests爬虫IP连接初始化问题及解决方案
  • Flutter 中数据存储的四种方式
  • Vue 路由缓存 防止路由切换数据丢失 路由的生命周期
  • 使用 Splashtop 的开放 API 简化 IT 工作流程
  • 大一,小小练习题--含答案
  • JAVA必应回答。
  • 2.发送邮件+开发注册功能
  • 【140天】尚学堂高淇Java300集视频精华笔记(86-87)
  • Bytom交易说明(账户管理模式)
  • Facebook AccountKit 接入的坑点
  • iOS 系统授权开发
  • Javascript基础之Array数组API
  • Java知识点总结(JavaIO-打印流)
  • macOS 中 shell 创建文件夹及文件并 VS Code 打开
  • mongodb--安装和初步使用教程
  • PhantomJS 安装
  • 纯 javascript 半自动式下滑一定高度,导航栏固定
  • 诡异!React stopPropagation失灵
  • 回顾2016
  • 入手阿里云新服务器的部署NODE
  • 数据库写操作弃用“SELECT ... FOR UPDATE”解决方案
  • 算法系列——算法入门之递归分而治之思想的实现
  • 体验javascript之美-第五课 匿名函数自执行和闭包是一回事儿吗?
  • 最简单的无缝轮播
  • 带你开发类似Pokemon Go的AR游戏
  • 曜石科技宣布获得千万级天使轮投资,全方面布局电竞产业链 ...
  • #[Composer学习笔记]Part1:安装composer并通过composer创建一个项目
  • #git 撤消对文件的更改
  • (06)金属布线——为半导体注入生命的连接
  • (C#)一个最简单的链表类
  • (C)一些题4
  • (cos^2 X)的定积分,求积分 ∫sin^2(x) dx
  • (附源码)计算机毕业设计高校学生选课系统
  • (黑马出品_高级篇_01)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式
  • (转)ObjectiveC 深浅拷贝学习
  • **PyTorch月学习计划 - 第一周;第6-7天: 自动梯度(Autograd)**
  • **登录+JWT+异常处理+拦截器+ThreadLocal-开发思想与代码实现**
  • ./和../以及/和~之间的区别
  • .java 指数平滑_转载:二次指数平滑法求预测值的Java代码
  • .net on S60 ---- Net60 1.1发布 支持VS2008以及新的特性
  • .NET/C# 在代码中测量代码执行耗时的建议(比较系统性能计数器和系统时间)
  • .Net程序猿乐Android发展---(10)框架布局FrameLayout
  • /bin/bash^M: bad interpreter: No such file or directory
  • [ Linux ] git工具的基本使用(仓库的构建,提交)
  • [ vulhub漏洞复现篇 ] Celery <4.0 Redis未授权访问+Pickle反序列化利用
  • [1525]字符统计2 (哈希)SDUT
  • [2009][note]构成理想导体超材料的有源THz欺骗表面等离子激元开关——