单向无头链表实现

目录

1. 为什么要有链表?

2. 链表的种类

 3. 具体功能实现

(1)节点结构体定义

(2)申请节点

(3)尾插

(4)尾删

(5)头插

(6)头删

(7)查找

(8)指定位置插入

(9)插入指定位置后面

(10)消除指定位置元素

(11)消除指定位置后的数据

(12)打印链表

(13)销毁链表

4. 完整代码

1.  slist.c

2.  slist.h

 3. 测试文件


1. 为什么要有链表?

上篇文章我们介绍了顺序表,顺序表有很多缺陷比如

1. 空间不够需要增容,增容需要付出代价

2. 为了避免频繁扩容,我们满了基本上都是扩2倍,可能会导致一定的空间浪费

3. 要求数据从开始位置连续存储,我们在头部或中间插入删除数据,需要挪动数据,效率不高

 那有人就针对顺序表的诸多缺陷,设计出了链表

链表的优缺点如下:

优点:
按需申请空间,不用了就释放空间,头部中间插入删除数据不需要挪动数据,不存在空间浪费。

缺点:

每存一个数据都要存一个指针去链接后面的内存节点,不支持随机访问(用下标直接访问第i个)

2. 链表的种类

单向或双向,带头或不带头,循环或不循环

共八种

最常用的为

1. 无头单向非循环链表

结构简单,更多作为其他数据结构的子结构

2. 带头双向循环链表 

结构复杂,一般用于单独存储数据,实际使用中结构复杂实现简单

 3. 具体功能实现

(1)节点结构体定义
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<assert.h>
typedef int SLTDataType;typedef struct SListNode
{int data;struct SListNode* next;}SListNode;

data 用来存储数据

next用来存储下一个节点的地址

(2)申请节点

插入数据时节省代码量

SListNode* BuySListNode(SLTDataType x)
{SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));if (newnode == NULL){exit(-1);}newnode->data = x;newnode->next = NULL;return newnode;
}
(3)尾插

在链表的末尾链接上一个节点

void SListPushBack(SListNode** phead, SLTDataType x)
{	 SListNode* newnode = BuySListNode(x);if (*phead == NULL){*phead = newnode;}else{SListNode* tail = *phead;while (tail->next != NULL){tail = tail->next;}tail->next = newnode;}}
(4)尾删

删除链表最末尾的那个节点

void SListPopBack(SListNode** phead)
{	if (*phead == NULL)return;//assert(*phead!=NULL);if ((*phead)->next == NULL){free(*phead);*phead = NULL;}else{SListNode* tail = *phead;SListNode* t = tail;while (tail->next != NULL){t = tail;tail = tail->next;}free(tail);tail = NULL;t->next = NULL;//SListNode* tail = *phead;//while (tail->next->next != NULL)//{//	tail = tail->next;//}//free(tail->next);//tail->next = NULL;}}
(5)头插

在链表头部插入数据

void SListPushFront(SListNode** phead, SLTDataType x)
{SListNode* newnode = BuySListNode(x);newnode->next = *phead;*phead = newnode;
}
(6)头删

删除链表头部的数据

void SListPopFront(SListNode** phead)
{if (*phead == NULL)return;SListNode* front=(*phead)->next;free(*phead);*phead = front;
}
(7)查找

查找链表中的某个数据,返回地址,找不到返回空指针

SListNode* SListFind(SListNode* phead, SLTDataType x)
{SListNode* cur = phead;while (cur){if (cur->data == x){return cur;}else{cur = cur->next;}}return  NULL;
}
(8)指定位置插入

根据给的地址插入数据(插入到指定位置前面)

void SListInsert(SListNode** phead, SListNode* pos, SLTDataType x)//pos配合查找函数
{SListNode* newnode = BuySListNode(x);if (*phead == pos){void SListPopFront(phead);}//找到pos的前一位置else{ SListNode* posPrey = *phead;while (posPrey->next != pos){posPrey = posPrey->next;}posPrey->next = newnode;newnode->next = pos;}
}
(9)插入指定位置后面
void SListInsertTail(SListNode* pos, SLTDataType x)
{assert(pos);SListNode* newnode = BuySListNode(x);newnode->next = pos->next;pos->next = newnode;
}
(10)消除指定位置元素
void SListErase(SListNode** phead, SListNode* pos)
{assert(head && (*head));assert(pos);if (*phead == pos){*phead = (*phead)->next;free(pos);pos = NULL;}else{SListNode* prev = *phead;while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);pos = NULL;}
}
(11)消除指定位置后的数据
void SListEraseAfter(SListNode* pos)
{assert(pos);if (pos == NULL){exit(-1);}SListNode* next = pos->next;pos->next = next->next;free(next);next = NULL;
}
(12)打印链表
void SListPrint(SListNode* phead)
{SListNode* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("\n");
}

(13)销毁链表

使用结束后要销毁

void SListDestroy(SListNode** phead)
{assert(phead && *phead);SListNode* plist = *phead;while (plist->next != NULL){plist = plist->next;free(*phead);*phead = plist;}free(*phead);*phead = NULL;}

4. 完整代码

1.  slist.c
#define _CRT_SECURE_NO_WARNINGS h
#include"sllist.h"SListNode* BuySListNode(SLTDataType x)
{SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));if (newnode == NULL){exit(-1);}newnode->data = x;newnode->next = NULL;return newnode;
}void SListPrint(SListNode* phead)
{SListNode* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("\n");
}void SListPushBack(SListNode** phead, SLTDataType x)
{	 SListNode* newnode = BuySListNode(x);if (*phead == NULL){*phead = newnode;}else{SListNode* tail = *phead;while (tail->next != NULL){tail = tail->next;}tail->next = newnode;}}void SListPushFront(SListNode** phead, SLTDataType x)
{SListNode* newnode = BuySListNode(x);newnode->next = *phead;*phead = newnode;
}void SListPopBack(SListNode** phead)
{	if (*phead == NULL)return;//assert(*phead!=NULL);if ((*phead)->next == NULL){free(*phead);*phead = NULL;}else{SListNode* tail = *phead;SListNode* t = tail;while (tail->next != NULL){t = tail;tail = tail->next;}free(tail);tail = NULL;t->next = NULL;//SListNode* tail = *phead;//while (tail->next->next != NULL)//{//	tail = tail->next;//}//free(tail->next);//tail->next = NULL;}}void SListPopFront(SListNode** phead)
{if (*phead == NULL)return;SListNode* front=(*phead)->next;free(*phead);*phead = front;
}SListNode* SListFind(SListNode* phead, SLTDataType x)
{SListNode* cur = phead;while (cur){if (cur->data == x){return cur;}else{cur = cur->next;}}return  NULL;
}
void SListInsert(SListNode** phead, SListNode* pos, SLTDataType x)//pos配合查找函数
{SListNode* newnode = BuySListNode(x);if (*phead == pos){void SListPopFront(phead);}//找到pos的前一位置else{ SListNode* posPrey = *phead;while (posPrey->next != pos){posPrey = posPrey->next;}posPrey->next = newnode;newnode->next = pos;}
}
void SListInsertTail(SListNode* pos, SLTDataType x)
{assert(pos);SListNode* newnode = BuySListNode(x);newnode->next = pos->next;pos->next = newnode;
}
void SListErase(SListNode** phead, SListNode* pos)
{assert(head && (*head));assert(pos);if (*phead == pos){*phead = (*phead)->next;free(pos);pos = NULL;}else{SListNode* prev = *phead;while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);pos = NULL;}
}
void SListEraseAfter(SListNode* pos)
{assert(pos);if (pos == NULL){exit(-1);}SListNode* next = pos->next;pos->next = next->next;free(next);next = NULL;
}
void SListDestroy(SListNode** phead)
{assert(phead && *phead);SListNode* plist = *phead;while (plist->next != NULL){plist = plist->next;free(*phead);*phead = plist;}free(*phead);*phead = NULL;}
2.  slist.h
#define _CRT_SECURE_NO_WARNINGS h
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<assert.h>
typedef int SLTDataType;typedef struct SListNode
{int data;struct SListNode* next;}SListNode;void SListPrint(SListNode* phead);
//打印
void SListPushBack(SListNode** phead, SLTDataType x);
//尾插
void SListPushFront(SListNode** phead,SLTDataType x);
//头插
void SListPopBack(SListNode** phead);
//尾删
void SListPopFront(SListNode** phead);
//头删
SListNode* SListFind(SListNode* phead, SLTDataType x);
//找位置
//在pos位置之前去插入一个节点
void SListInsert(SListNode** phead, SListNode* pos, SLTDataType x);
//在pos前面插入//void SListInsert(SListNode* phead, int pos, SLTDataType x);void SListErase(SListNode** phead, SListNode* pos);
//找坐标删除
void SListEraseAfter(SListNode* pos);
//删除坐标后面的值void SListDestroy(SListNode** phead)
 3. 测试文件
#define _CRT_SECURE_NO_WARNINGS h
#include"sllist.h"
#include<stdio.h>
#include<stdlib.h>
void TestSList1()
{SListNode* plist = NULL;//初始值SListPushBack(&plist, 1);SListPushBack(&plist, 2);SListPushBack(&plist, 3);SListPushFront(&plist, 4);SListPushFront(&plist, 5);SListPrint(plist);SListPopFront(&plist);SListPopFront(&plist);SListPrint(plist);
}
void TestSList2()
{SListNode* plist = NULL;//初始值SListPushBack(&plist, 1);SListPushBack(&plist, 2);SListPushBack(&plist, 3);SListPushBack(&plist, 3);SListPushBack(&plist, 2);SListNode* pos = SListFind(plist, 2);SListInsert(&plist ,pos,20);SListPrint(plist);//SListDestroy(&plist);//SListPrint(plist);int i = 1;while (pos){printf("第%d个节点%p->%d\n", i++, pos, pos->data);pos = SListFind(pos->next, 2);}pos = SListFind(plist, 3);while(pos){pos->data = 30;pos = SListFind(pos->next, 3);}SListPrint(plist);
}
typedef struct xx
{int* n;
}xx;
int main()
{TestSList2();//xx p;//int a = 0;//p.n = &a;//a = 2;//printf("%d", *(p.n));return 0;
}

这篇文章就到这里了,希望可以帮到您

(๑′ᴗ‵๑)I Lᵒᵛᵉᵧₒᵤ❤

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

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

相关文章

SDUT 链表3

7-3 sdut-C语言实验-链表的结点插入 分数 20 全屏浏览 切换布局 作者 马新娟 单位 山东理工大学 给出一个只有头指针的链表和 n 次操作&#xff0c;每次操作为在链表的第 m 个元素后面插入一个新元素x。若m 大于链表的元素总数则将x放在链表的最后。 输入格式: 多组输入。…

【Python设计模式13】抽象工厂模式

抽象工厂模式&#xff08;Abstract Factory Pattern&#xff09;是一种创建型设计模式&#xff0c;它提供一个接口&#xff0c;用于创建一系列相关或依赖的对象&#xff0c;而无需指定它们具体的类。抽象工厂模式通过对产品类的抽象&#xff0c;使客户端可以使用抽象工厂来创建…

算法和远程编程题

文章目录 2024心得0504 堆/栈/队列用两个栈实现队列 05 哈希BM52 数组中只出现一次的两个数字 2024 牛客网在线编程 心得 大厂会要求&#xff0c;这个其实没有什么用&#xff0c;就是故意为难人&#xff0c;但是要想找一份工作&#xff0c;这个还是只能去遵守规则。面试造火…

go语言之函数基础

1.介绍 函数是基本的代码块&#xff0c;Go是编译型语言&#xff0c;所以函数编写的顺序是无关紧要的&#xff0c;但是我们一般把main&#xff08;&#xff09;函数写在文件的前面&#xff0c;其他函数按照一定的逻辑顺序编写&#xff08;例如函数被调用顺序&#xff09;。 编写…

PHP的多样化执行方式(parallel PHP多线程实现,原生协程实现,多进程实现,ZTS、NTS、TS又是什么)

进程、线程、协程 进程&#xff1a;应用程序的启动实例&#xff0c;运行起的代码叫进程&#xff0c;有独立的内存空间&#xff0c;类比工厂的P个&#xff08;P1单进程&#xff0c;P>1多进程&#xff09;车间。线程&#xff1a;线程是CPU调度的最小单位&#xff0c;是进程内…

面向对象-----继承

前面向大家介绍了面向对象中的封装性&#xff0c;今天再来向大家介绍面向对象的继承和多态的两大特性。 1.继承 1.1 为什么需要继承&#xff1f; 在java语言中&#xff0c;我们用类来描述世间万物&#xff0c;虽然万物非常复杂&#xff0c;但总有一些共同点&#xff0c;如果…

CAD2023 2024 2025以上版本出现无法运行 AutoCAD,原因可能如下1) 此版本的 AutoCAD 安装不正确

错误提示如下 此版本的 AutoCAD 安装不正确 缺少依赖组件Microsoft Edge webview2 Runtime 缺少依赖组件 Microsoft.NET跟You must install .NET Desktop Runtime 打开autoremove&#xff0c;点击扩展&#xff0c;输入 无法运行&#xff0c;点击搜索 你的软件属于什么版本…

fork 与 vfork 的区别

关键区别一&#xff1a; vfork 直接使用父进程存储空间&#xff0c;不拷贝。 关键区别二&#xff1a; vfork保证子进程先运行,当子进程调用exit退出后&#xff0c;父进程才执行。 我们可以定义一个cnt&#xff0c;在子进程中让它变成3&#xff0c; 如果使用fork&#xff0c;那…

PHP 7.4开始 调用方法success()可根据参数名称来指定参数值uccess(data: $list, count: $count)

定义了方法&#xff1a; static function success(int $code200,string $msgok,array $data[],int $count0,int $pages1){return [code>$code,msg>$msg,data>$data,count>$count,pages >$pages]; } 引用&#xff1a;(不用按顺序) return self::success(data:…

【LeetCode】【1】两数之和(1141字)

文章目录 [toc]题目描述样例输入输出与解释样例1样例2样例3 提示进阶Python实现哈希表 个人主页&#xff1a;丷从心 系列专栏&#xff1a;LeetCode 刷题指南&#xff1a;LeetCode刷题指南 题目描述 给定一个整数数组nums和一个整数目标值target&#xff0c;请在该数组中找出…

【Vue3进阶】- Vite 配置

视频 Vite 配置 基础配置项 server 选项 本地运行时&#xff0c;开发环境服务器的配置。 host 默认 localhost&#xff0c;设置为 true 或 0.0.0.0 时会监听所有地址&#xff0c;让局域网内的设备进行访问&#xff0c;否则只有本机能访问项目。 open 默认 false&#x…

C#如何通过反射获取外部dll的函数

在C#中&#xff0c;你可以使用反射&#xff08;Reflection&#xff09;来加载外部的DLL&#xff08;动态链接库&#xff09;并获取其中的函数&#xff08;在C#中通常称为方法&#xff09;。但是&#xff0c;请注意&#xff0c;反射主要用于访问类型信息&#xff0c;并且对于非托…

Apache 平滑升级:逐步指南与示例

在维护Web服务器时&#xff0c;Apache的平滑升级是一项重要的操作&#xff0c;它可以确保服务在升级过程中继续对外提供服务&#xff0c;最小化或无需停机时间。本文将详细介绍如何在Linux系统中平滑升级Apache HTTP服务器&#xff0c;提供一个完整的操作示例&#xff0c;并说明…

Typora+PicGo+Gitee设置图床,解决CSDN上传markdown文件图片加载不出来的问题(超级实用)

注&#xff1a; 由于gitee现在已经加上了防盗链&#xff0c;并且只支持1M的图片&#xff0c;我觉得不是很好用&#xff08;可以买腾讯云或阿里云等&#xff09;&#xff0c;之后找到比较好点的图床工具也会持续更新的。&#xff08;sm.ms好像还好&#xff0c;github网速不太稳定…

计算机网络 3.5局域网

第五节 局域网 一、认识局域网 1.定义&#xff1a; ①功能性&#xff1a;一组在地理位置上相隔不远的计算机及其设备按照一定的连接方式组织起来的&#xff0c;以实现用户间相互通信的资源共享的网络系统。 ②技术性&#xff1a;由特定类型的传输媒体和网络适配器互联在一起…

基于Vue和uni-app的增强型单选ccRadioView组件开发

标题&#xff1a;基于Vue和uni-app的增强单选组件ccRadioView的设计与实现 摘要&#xff1a;本文将详细介绍如何使用Vue和uni-app构建一个简单、好用且通用的单选框组件ccRadioView。该组件提供了单选列表的功能&#xff0c;并支持反向传值&#xff0c;方便开发者快速实现单选…

太速科技-基于5VLX110T FPGA FMC接口功能验证6U CPCI平台

基于5VLX110T FPGA FMC接口功能验证6U CPCI平台 一、板卡概述   本板卡是Xilinx公司芯片V5系列芯片设计信号处理板卡。由一片Xilinx公司的XC5VLX110T-1FF1136 / XC5VSX95T-1FF1136 / XC5VFX70T-1FF1136芯片组成。FPGA接1片DDR2内存条 2GB&#xff0c;32MB Nor flash存储器&a…

【Java基础】IO流(4) —— 转换流、打印流

【Java基础】IO流(1) —— 简介 【Java基础】IO流(2) —— 字符流 【Java基础】IO流(3) —— 字节流 【Java基础】IO流(4) —— 转换流、打印流 【Java基础】IO流(5) —— 序列流、内存流 【Java基础】IO流(6) —— 随机访问文件流、数据流 转换流 InputStreamReader 是字节输…

【Linux】在Linux中使用date命令

路过了学校花店 荒野到海边 有一种浪漫的爱 是浪费时间 徘徊到繁华世界 才发现你背影 平凡得特别 绕过了城外边界 还是没告别 爱错过了太久 反而错得完美无缺 幸福兜了一个圈 想去的终点 就在原点 &#x1f3b5; 林宥嘉《兜圈》 前言 date命令是Linux系统…