数据结构--单链表

news2025/1/19 19:25:02

1.定义

由于顺序表的插入删除操作需要移动大量的元素,影响了运行效率,因此引入了线性表的链式存储——单链表。单链表通过一组任意的存储单元来存储线性表中的数据元素,不需要使用地址连续的存储单元,因此它不要求在逻辑上相邻的两个元素在物理位置上也相邻。

2.特点

  1. 单链表不要求逻辑上相邻的两个元素在物理位置上也相邻,因此不需要连续的存储空间。
  2. 单链表是非随机的存储结构,即不能直接找到表中某个特定的结点。查找某个特定的结点时,需要从表头开始遍历,依次查找。
  3. 对于每个链表结点,除了存放元素自身的信息外,还需要存放一个指向其后继的指针。

3.代码

1.lishh

//头文件 ".h",存放结构或者函数的声明
//带头结点单链表,逻辑相邻,物理不一定相邻,为了找到下一个节点,就必须增加下一个节点的地址
//尾节点:最后一个节点,在单链表中,尾节点的next为NULL,NULL是单链表的结尾标志
//头结点:其数据域不使用或者存放链表长度.其作用,相对于一个标杆,简化操作

#pragma once

typedef int ElemType;

typedef struct Node
{
	ElemType data;//数据
	struct Node* next;//后继指针
}Node,*List;//List == Node *

//typedef Node* List;//List == Node *
//List本质是Node*,但含义不同,List表示一条链表,Node*表示一个节点的地址

//初始化plist
void InitList(List plist);

//书上的写法
void InitList(List *pplist);

//往plist中头部插入数字val
bool Insert_head(List plist, ElemType val);

//往plist中的尾部插入数字val
bool Insert_tail(List plist , ElemType val);

//在plist中查找val值,找到返回该节点地址,失败返回NULL
Node* Search(List plist, ElemType val);

//删除plist中的第一个val
bool DeleteVal(List plist, ElemType val);

//判断plist是否为空链表(没有数据节点)
bool IsEmpty(List plist);

//获取plist长度,数据节点的个数
int GetLength(List plist);

//获取plist链表的pos位置的值
//int GetElem(List plist, int pos);//设计有问题:无法清晰的表示出错
bool GetElem(List plist,int pos,int *rtval);//rtval:输出参数

//获取val的前驱
Node* Prior(List plist, ElemType val);

//获取val的后继
Node* Next(List plist, ElemType val);

//输出plist的所有数据
void Show(List plist);

//清空数据
void Clear(List plist);

//销毁
void Destroy(List plist);

//逆置,考试非常多
void Reverse(List plist);

2.list.cpp

1.初始化plist

通常会用头指针来标识一个单链表,头指针为NULL时表示一个空表。但是,为了操作方便,会在单链表的第一个结点之前附加一个结点,称为头结点。头结点的数据域可以不设任何信息,也可以记录表长等信息。头结点的指针域指向线性表的第一个元素结点。

空的带头节点的单链表

含有三个结点的带头结点的点链表 

头结点和头指针的区分:不管带不带头结点,头指针始终指向单链表的第一个结点,而头结点是带头结点的单链表中的第一个结点,结点内通常不存储信息。
那么单链表的初始化操作就是申请一个头结点,将指针域置空。

void InitList(List plist)
{
	assert(plist != NULL);

	//头结点的数据不使用 plist->data可以不处理
	plist->next = NULL;
}

2.书上的写法

void InitList(List* pplist)
{
	assert(pplist != NULL);
	*pplist = (Node*)malloc(sizeof(Node));//动态创建头结点
	assert(*pplist != NULL);
	if (*pplist == NULL)
		return;
	(*pplist)->next = NULL;
}

3.往plist中头部插入数字val

所谓头插法建立单链表是说将新结点插入到当前链表的表头,即头结点之后。如图所示:

 算法思想:首先初始化一个单链表,其头结点为空,然后循环插入新结点*s:将s的next指向头结点的下一个结点,然后将头结点的next指向s。

bool Insert_head(List plist, ElemType val)//O(1)
{
	//1.动态创建一个新节点
	Node* p = (Node*)malloc(sizeof(Node));
	assert(p != NULL);
	if (p == NULL)
		return false;
	//2.把数据val存放到新节点
	p->data = val;

	//3.把新节点插入在头结点的后面
	p->next = plist->next;
	plist->next = p;

	return true;
}

4.往plist中的尾部插入数字val

需要指出的是,头插法建立的单链表中结点的次序和输入数据的顺序不一致,是相反的。若希望两者的顺序是一致的,则可采用尾插法建立单链表。

所谓尾插法建立单链表,就是将新结点插入到当前链表的表尾。如下图所示:

 最后情况插入元素a[n-1],需将终结点的指针域置空

bool Insert_tail(List plist, ElemType val)//O(n)
{
	//1.动态创建新节点
	Node* p = (Node*)malloc(sizeof(Node));
	assert(p != NULL);
	if (p == NULL)
		return false;
	//	2.把值存放到新节点
	p->data = val;//p->next = NULL;
	//	3.找到链表尾巴
	Node* q;
	for (q = plist; q->next != NULL; q = q->next)
		;

	//	4.把新节点插在尾节点的后面,把p插入在q的后面
	p->next = q->next;//p->next = NULL;
	q->next = p;

	return true;
}

5.在plist中查找val值,找到返回该节点地址,失败返回NULL

 查找值val在单链表L中的结点指针。

 算法思想:从单链表的第一个结点开始,依次比较表中各个结点的数据域的值,若某结点数据域的值等于val,则返回该结点的指针;若整个单链表中没有这样的结点,则返回空。

Node* Search(List plist, ElemType val)
{
	for (Node* p = plist->next; p != NULL; p = p->next)
	{
		if (p->data == val)//找到了
			return p;
	}
	return NULL;//没有找到
}

6.删除plist中的第一个val

算法思想:先检查删除位置的合法性,然后从头开始遍历,找到表中val结点,即被删除结点的前驱结点*p,被删除结点为*q,修改*p的指针域,将其指向*q的下一个结点,最后再释放结点*q的存储空间。

bool DeleteVal(List plist, ElemType val)
{
	//1.找val的前驱
	Node* p = Prior(plist,val);//指向前驱节点
	if (p == NULL)//没有val
		return false;

	//free(p->next);//错误
	//p->next = p->next->next;
	Node* q = p->next;//标记需要删除的节点
	p->next = q->next;//p->next = p->next->next;将q从链表中剔除
	free(q);//释放节点

	return true;
	
}

7.判断plist是否为空链表(没有数据节点)

算法思想:要判断带头结点的单链表是否为空,只需要看头结点的指针域即可,如果头结点的指针域为空,即单链表中只有一个头结点,那么该单链表为空表。

bool IsEmpty(List plist)
{
	return plist->next == NULL;//等同下面的if  else

	/*if (plist->next == NULL)
		return true;
	else
		return false;*/
}

8.获取plist长度,数据节点的个数

算法思想:声明一个指针p,p指向头结点指向的第一个结点,如果p指向的结点不为空,那么长度加一,将p指向下一个结点,直到遍历到最后一个结点为止。

int GetLength(List plist)
{
	int count = 0;//计数器

	for (Node* p = plist->next; p != NULL; p = p->next)
		count++;

	return count;
}

9.获取plist链表的pos位置的值

//int GetElem(List plist, int pos)//设计有问题
//{
//	if (pos < 0 || pos >= GetLength(plist))//下标不存在
//		return -1;//不可以,可能和正常的值冲突
//}
bool GetElem(List plist, int pos, int* rtval)//rtval:输出参数
{
	if (pos < 0 || pos >= GetLength(plist))//出错
		return false;

	Node* p=plist->next;
	for (int i=0;  i < pos; p = p->next, i++)
	{
		;
	}

	*rtval = p->data;

	return true;
}

10.获取val的前驱

Node* Prior(List plist, ElemType val)
{
	for (Node* p = plist; p != NULL; p = p->next)//bug  
	{
		if (p->next->data == val)//找到了
			return p;
	}
	return NULL;//失败了
}

11.获取val的后继

Node* Next(List plist, ElemType val)
{
	for (Node* p = plist->next; p != NULL; p = p->next)
	{
		if (p->data == val)
			return p->next;
	}

	return NULL;
}

12.输出plist的所有数据

算法思想:声明一个指针p,从头结点指向的第一个结点开始,如果p不为空,那么就输出当前结点的值,并将p指向下一个结点,直到遍历到最后一个结点为止。

void Show(List plist)
{
	for (Node* p = plist->next; p != NULL; p = p->next)//遍历所有的数据节点
	{
		printf("%d ",p->data);
	}
	printf("\n");
}

13.清空数据

void Clear(List plist)
{
	Destroy(plist);
}

14.销毁

void Destroy1(List plist)
{
	if (plist == NULL || plist->next == NULL)
		return;
	Node* p = plist->next;
	Node* q;
	plist->next = NULL;
	while (p != NULL)
	{
		q = p->next;
		free(p);
		p = q;
	}
}

void Destroy(List plist)
{
	Node* p;//指向第一个数据节点
	while (plist->next != NULL)//还有数据结构
	{
		p = plist->next;
		plist->next = p->next;//剔除p
		free(p);
	}
}

3.test.cpp

//测试文件,测试用例
#include <stdio.h>
#include "list.h"
//#include <vld.h>//必须安装vld,如果没有安装不能使用

//实现两个集合的并集,A=AUB(相同的部分取一次,不同部分全取)
//例如A={1,2,3,4};B={5,3,1,6};结果={1,2,3,4,5,6};
void Merge(List plistA, List plistB)
{
	//算法:遍历plistB,查看当前值在plistA中是否存在,如果存在则什么都不做,
	//不存在则将值插入到plistA中
	int val;
	for (int i = 0; i < GetLength(plistB); i++)
	{
		GetElem(plistB,i,&val);//获取plistB,i下标的值
		if (Search(plistA, val) == NULL)//不存在
		{
			Insert_tail(plistA,val);
		}
	}
}

int main()
{
	Node headA;
	Node headB;
	InitList(&headA);
	InitList(&headB);
	Insert_tail(&headA, 1);
	Insert_tail(&headA, 2);
	Insert_tail(&headA, 3);
	Insert_tail(&headA, 4);
	Show(&headA);

	Insert_tail(&headB, 5);
	Insert_tail(&headB, 3);
	Insert_tail(&headB, 1);
	Insert_tail(&headB, 6);
	Show(&headB);

	Merge(&headA,&headB);
	printf("合并后的值\n");
	Show(&headA);
	Show(&headB);

	Destroy(&headA);
	Destroy(&headB);

	return 0;
}

结果:

 

 

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

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

相关文章

携手!Kyligence 支持 Amazon EMR Serverless,赋能云上企业降本增效

近日&#xff0c;Kyligence 与亚马逊云科技宣布 Kyligence Cloud 4.6 正式支持 Amazon EMR Serverless&#xff0c;共同助力企业构建高效、低成本云上数据分析。今天&#xff0c;我们将为大家介绍 Kyligence Cloud 4.6 如何通过集成 Amazon EMR Serverless 提供端到端的云上数…

字节给我狠狠上了一课:危机来的时候你连准备时间都没有~

各大互联网公司的接连裁员&#xff0c;政策限制的行业接连消失&#xff0c;让今年的求职雪上加霜&#xff0c;想躺平却没有资本&#xff0c;还有人说软件测试岗位饱和了&#xff0c;对此很多求职者深信不疑&#xff0c;因为投出去的简历回复的越来越少了。甚至还有不少大厂直接…

shell的test命令和两种参数获取方式,详解getopts以及eval、exec、export、read、shift内置命令

一、linux里while [-n “$1”]这里的-n是什么意思? -n str&#xff0c;字符串不为null&#xff0c;长度大于零while [ -n “$1” ] 第一个参数不为空&#xff0c;返回TRUE执行while循环do … done 二、Shell test 命令 Shell中的 test 命令用于检查某个条件是否成立&#x…

【web渗透思路】敏感信息泄露(网站+用户+服务器)

目录 一、信息泄露示例 1、示例&#xff1a; 二、泄露方式 1、原理&#xff1a; 三、泄露危害 1、危害&#xff1a; 四、泄露挖掘 1、爬虫文件 2、目录信息 3、越权访问 4、开发注释、js文件 5、错误提示 6、调试信息 7、备份等目录文件 8、配置不安全 9、版本控…

数字集成电路设计(五、仿真验证与 Testbench 编写)(五)

文章目录9. 编译预处理语句9.1 仿真时间标度10. Verilog HDL测试方法简介9. 编译预处理语句 这些在C语言中都有涉及&#xff0c;但是在硬件描述语言中相对用的不是特别多&#xff0c;比如说include&#xff0c;在语言用的特别多&#xff0c;但是在硬件描述语言很少include&…

【每天学习一点新知识】OWSAP TOP10

OWASP OWASP开放式Web应用程序安全项目&#xff08;open web application security project&#xff09;每年会通过确定企业面临的最严重的10类威胁&#xff0c;以此提高人们对Web应用程序安全的关注度。 2021年 A01 失效的访问控制 未对通过身份验证的用户实施恰当的访问控…

裸辞4个月,面试了30家公司,终于找到理想工作了

上半年裁员&#xff0c;下半年裸辞&#xff0c;有不少人高呼裸辞后躺平真的好快乐&#xff01;但也有很多人&#xff0c;裸辞后的生活五味杂陈。 面试30次终于找到心仪工作 因为工作压力大、领导PUA等各种原因&#xff0c;今年2月下旬我从一家互联网小厂裸辞&#xff0c;没想…

APS高级排产在冶金行业的应用

冶金工业是指对金属矿物的勘探、开采、精选、冶炼、以及轧制成材的工业部门。包括黑色冶金工业(即钢铁工业)和有色冶金工业两大类。随着业务量的不断扩大&#xff0c;仅具有传统的ERP系统和人工生产调度已逐渐不能满足精益生产改进的要求。建立高效的精益生产计划模型及其对应的…

Nginx模块开发之http handler实现流量统计(入门篇)

Nginx模块开发之http handler实现流量统计一、Nginx模块之http handler简介二、Nginx handler模块开发2.1、示例代码2.2、编写config文件2.3、编译模块到Nginx源码中2.4、修改conf文件2.5、执行效果三、Nginx的热更新总结后言一、Nginx模块之http handler简介 当nginx解析conf…

计算机毕业设计之java+ssm手机综合类门户网站

项目介绍 手机综合类门户网站采用ssm框架和eclipse编辑器、MySQL数据库设计并实现的,主要包括系统手机评测管理模块、文章管理模块、手机新闻管理、所有评论管理、登录模块、和退出模块等多个模块。 管理员的登录模块&#xff1a;管理员登录系统对本系统其他管理模块进行管理。…

2016-2021年各省高考分数线

2016-2021年各省高考分数线 包含各省市不同招生类别的高考分数线&#xff0c;包括普通本科、特殊类型招生、艺术类本科、体育类本科、普通专科的分数线。 普通高等学校招生全国统一考试&#xff08;Nationwide Unified Examination for Admissions to General Universities a…

网络防火墙入门

防火墙的作用&#xff1f; 防止从外向内的网络入侵行为的策略。后来&#xff0c;防火墙不但用于防范外网&#xff0c;例如&#xff1a;对企业内网的 DoS 攻击或非法访问等&#xff0c;也开始防范从内部网络向互联网泄露信息、把内部网络作为攻击跳板等行为。 什么是代理服务器…

Hash表(哈希表、散列表)

哈希表 概念 为什么需要哈希表 静态查找表与动态查找表中&#xff0c;为了查找某关键字值等于某个值的记录&#xff0c;都要经过一系列的关键字进行比较&#xff0c;以确定待查记录的储存位置或查找失败&#xff0c;查找的时间总是与比较次数有关 什么是哈希表 哈希表&…

[附源码]java毕业设计校园征兵及退役复原管理系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

视听杂志视听杂志社视听编辑部2022年第11期目录

专题:对外传播 共塑与去精英化&#xff1a;国家形象建构的实践创新——基于纪录片《柴米油盐之上》的多模态分析 董星雨;程欣;刘苏情; 3-7《视听》投稿&#xff1a;cnqikantg126.com 网络赋权下抖音国际版TikTok的海外传播策略探析 吴梦玲; 7-10 以画对话&#xff…

m3u8 文件格式详解

简介 M3U8 是 Unicode 版本的 M3U&#xff0c;用 UTF-8 编码。"M3U" 和 "M3U8" 文件都是苹果公司使用的 HTTP Live Streaming&#xff08;HLS&#xff09; 协议格式的基础&#xff0c;这种协议格式可以在 iPhone 和 Macbook 等设备播放。上述文字定义来自于…

一文玩转Java 泛型知识

✅作者简介&#xff1a;热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏&#xff1a;前端开发者…

青岛品质水稻共养 国稻种芯·中国水稻节:山东西海岸收获季

青岛品质水稻共养 国稻种芯中国水稻节&#xff1a;山东西海岸收获季 半岛全媒体记者 孟达 新闻中国采编网 中国新闻采编网 谋定研究中国智库网 中国农民丰收节国际贸易促进会 国稻种芯中国水稻节 中国三农智库网-功能性农业农业大健康大会报道&#xff1a;山东青岛西海岸新区王…

发了3000个短视频作品才总结出的9点快速破播放的技巧

大家好&#xff0c;我是我赢助手&#xff0c;专注于自媒体短视频去水印、去重和文案提取运营。 今天给大家分享下发了3000个短视频作品才总结出的9点快速破播放的技巧&#xff1a; 1、前期养号: 新号创建前7天不要急着发作品&#xff0c;刷兴趣标签养号&#xff0c;能让账号…

【机器学习】数据驱动方法在电网稳定分析应用浅谈

目录 一、数据驱动概述 二、数据驱动特点 三、数据驱动与其他方法对比 四、总结 五、参考文献 一、数据驱动概述 数据驱动在电力系统稳定分析中的应用&#xff0c;主要目标是从电网运行数据角度建立电力系统稳定分析模型&#xff0c;以数据之间的关联性分析视角挖掘电力系…