带头结点和尾指针的循环单链表(C语言)

这篇具有很好参考价值的文章主要介绍了带头结点和尾指针的循环单链表(C语言)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

循环单链表的一些基本概念



头结点

头结点的定义

头结点是链表中的第一个结点,其数据域一般无意义(有时可存放链表长度等)。

头结点目的

统一链表的操作。使得空表时的操作不用特殊处理,简化了链表的操作。



尾指针

尾指针的定义

尾指针是指向链表尾结点的指针。

尾指针的作用

用来找到整个链表
减少时间复杂度。链表的插入删除操作一般在表尾进行操作,头指针因为一开始指向头结点,要找到尾结点的时间复杂度为O(n),而尾指针指向的就是尾结点,所以找到尾结点的时间复杂度为O(1)



循环单链表的相关声明

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>


struct Node;
typedef struct Node* PtrToNode;									
typedef PtrToNode CircList;			// 表示指向链表的指针
typedef PtrToNode Position;			// 表示指向结点的指针
typedef int ElemType;				// 数据类型

// 结点的定义
typedef struct Node {
	ElemType element;
	Position next;
}Node;


void InitList(CircList* pprear);
bool IsEmptyList(CircList rear);
Position CreateNode(ElemType elem);
Position GetElem(CircList rear, int pos);
void InsertElem(CircList* pprear, int pos, ElemType elem);
void PushFront(CircList* ppRear, ElemType elem);
void PushBack(CircList* ppRear, ElemType elem);
ElemType DeleteElem(CircList* ppRear, int pos);
ElemType PopFront(CircList* ppRear);
ElemType PopBack(CircList* ppRear);
void ModifyElem(CircList rear, int pos, ElemType elem);
int GetLength(CircList rear);
void ClearList(CircList* ppRear);
// 非通用型
Position FirstSameElem(CircList rear, int elem);
int FirstSameElemPos(CircList rear, int elem);
int RemoveFirstSameElem(CircList* ppRear, ElemType elem);
void HeadInsert(CircList* ppRear);
bool InputInteger(int* elem);
void Print(CircList L);


二级指针

二级指针是指向指针的指针**,用来在函数中改变指针的位置
在带有尾指针的循环单链表中尾指针始终是指向尾结点的,所以当尾结点发生改变时,如插入,删除等操作改变了尾结点,那么就要传入二级指针移动尾指针



循环单链表的定义

struct Node;							// 先定义结点的结构体,不创建模板
typedef struct Node* PtrToNode;									
typedef PtrToNode CircList;				// 表示指向链表的指针
typedef PtrToNode Position;				// 表示指向结点的指针
typedef int ElemType;					// 链表的数据类型

// 结点结构体的定义
typedef struct Node {
	ElemType element;
	Position next;
}Node;


循环单链表的基本通用操作

1、初始化循环单链表

因为是循环链表,初始化时只有一个头结点,所以尾指针指向的是头结点,头结点指向的还是头结点。

void InitList(CircList* ppRear)
{
	(*ppRear) = (CircList)malloc(sizeof(Node));			// 创建头结点并使尾指针指向头结点
	assert((*ppRear) != NULL);							// 如果头结点创建失败则终止程序
	(*ppRear)->next = (*ppRear);						// 使头结点的下一个还是头结点,形成循环
}


2、循环单链表是否为空

单独写个函数,为了方便后面做特殊情况判断。

bool IsEmptyList(CircList rear)
{
	return rear->next == rear;		// 如果头结点的下一个结点还是头结点那么就是空表
}


3、创建新结点

单独写个创建新结点的函数方便以后插入相关的操作使用。

Position CreateNode(ElemType elem)
{
	Position node = (Position)malloc(sizeof(Node));		// 创建新结点
	if (node == NULL)									// 创建失败则提示并终止程序
	{
		puts("Creation failed.");
		exit(EXIT_FAILURE);
	}
	node->element = elem;								// 为新节点赋值
	node->next = NULL;									// 不知道新节点的下一个位置,所以暂时指向NULL
	return node;										// 返回新结点地址
}


4、查找指定位置上的元素

查找元素时会有两个错误
1、表为空表
2、查找的位置超出范围
所以要有这两个错误的处理
查找元素时有些特殊情况
1、查找位置为0时,因为pos为0,pos<0与count<pos都不成立,所以会直接返回头结点的位置
2、查找位置为表尾时count=pos,cur为尾结点,然后会跳出while循环,这时就会返回尾结点
(rear为尾结点,尾结点下一个是头结点)

Position GetElem(CircList rear, int pos)
{
	if (pos < 0)									// 左越界判断
	{
		puts("Wrong position.");
		exit(EXIT_FAILURE);
	}
	Position cur = rear->next;						// 从头结点开始找
	int count = 0;									// 用来计数
	// 右越界判断:如果循环了一遍链表还没找到那就代表位置越界,则终止程序
	while (count < pos)											
	{
		cur = cur->next;
		count++;
		// 当找到头结点时,说明已经循环完一遍链表但还没找到,代表越界了
		if (cur == rear->next)
		{
			puts("Wrong position.");
			exit(EXIT_FAILURE);
		}
	}
	return cur;										// 返回位置结点
}


5、在指定位置上插入结点

插入结点需要知道插入位置的前驱结点,这样才能将新结点连接上链表,而上面写的查找指定位置的元素函数就能帮我们做到。

void InsertElem(CircList* ppRear, int pos, ElemType elem)
{
	Position prec = GetElem((*ppRear), pos - 1);				// 查找前驱结点目的是为了将新结点连接上链表
	Position newnode = CreateNode(elem);							// 创建新结点
	newnode->next = prec->next;									// 将新节点连接上链表
	prec->next = newnode;
	if (prec == *ppRear)							// 如果新节点的位置是在最后一位那么要移动尾指针指向新节点
		(*ppRear) = newnode;
}

6、在首部插入新结点

要注意:如果是空表的情况要移动一次尾指针

void PushFront(CircList* ppRear, ElemType elem)
{
	Position newNode = CreateNode(elem);							// 创建一个新节点
	newNode->next = (*ppRear)->next->next;						// 使新节点指向首元结点
	(*ppRear)->next->next = newNode;							// 使首结点指向新节点,新节点变为新的首元结点
	// 对空表插入后只有头结点和第一个结点,此时尾指针指向头结点,这时候需要将尾指针移动到第一个结点上
	if ((*ppRear)->next->next == *ppRear)
		*ppRear = (*ppRear)->next;
}

7、在尾部插入新结点

每次在尾部插入新结点时都要移动尾指针

void PushBack(CircList* ppRear, ElemType elem)
{
	Position newNode = CreateNode(elem);							// 创建一个新节点
	newNode->next = (*ppRear)->next;							// 使新节点指向首元结点
	(*ppRear)->next = newNode;									// 使尾结点指向新结点
	(*ppRear) = newNode;										// 移动尾指针
}

8、删除指定位置上的结点

要检查该链表是否为空表和越界问题
注意:当要删除的结点是尾结点的下一个时,GetElem函数不会报错,而是会返回尾结点,所以要做特殊越界处理

ElemType DeleteElem(CircList* ppRear, int pos)
{
	if (IsEmptyList(*ppRear))									// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	Position prec = GetElem(*ppRear, pos - 1);					// 找到将被删除的结点的前驱节点
	// 如果前驱结点为最后一个时GetElem函数不会报错,所以要做特殊处理
	if (prec == (*ppRear))										
	{
		puts("Wrong position.");
		exit(EXIT_FAILURE);
	}
	Position cur = prec->next;									// 标记要被删除的结点
	ElemType elem = cur->element;								// 标记要被删除的元素
	if (cur == (*ppRear))									// 如果被删除的结点是最后一个时就要移动尾指针指向他
		*ppRear = prec;
	prec->next = cur->next;										// 将前驱结点与被删除结点的后继结点连接在一起
	free(cur);													// 删除(释放)结点
	return elem;												// 返回被删除的元素
}

9、从头部删除元素

ElemType PopFront(CircList* ppRear)
{
	if (IsEmptyList(*ppRear))									// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	Position cur = (*ppRear)->next->next;						// 标记将被删除的首元结点
	ElemType elem = cur->element;								// 标记将被删除的元素
	(*ppRear)->next->next = cur->next;							// 将首元结点改为将被删除的结点的后一个
	if (cur == (*ppRear))									// 如果链表中只有一个结点,那么要将指针重新指向头结点
		*ppRear = (*ppRear)->next;
	free(cur);													// 删除(释放)结点
	return elem;											// 返回被删除的元素
}

10、从尾部删除结点

每次删除结点时都要找到尾结点的前驱结点,并且移动尾指针指向它

ElemType PopBack(CircList* ppRear)
{
	if (IsEmptyList(*ppRear))									// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	Position prec = (*ppRear)->next;							
	Position cur = (*ppRear);									// 标记将被删除的尾结点
	ElemType elem = (*ppRear)->element;
	while (prec->next != (*ppRear))								// 找到尾结点的前驱节点
		prec = prec->next;
	prec->next = (*ppRear)->next;								// 将尾结点的前驱结点与头结点连接变为新的尾结点
	*ppRear = prec;												// 使尾指针指向新的尾结点
	free(cur);													// 删除(释放)结点
	return elem;
}

11、修改特定位置上的元素

要做空表判断
pos为0时的结点为头结点,GetElem函数不会报错,所以要特殊处理

void ModifyElem(CircList rear, int pos, ElemType elem)
{
	if (IsEmptyList(rear))										// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	if (pos == 0)												// 左越界判断
	{
		puts("Wrong position.");
		exit(EXIT_FAILURE);
	}
	Position cur = GetElem(rear, pos);							// 找到该结点
	cur->element = elem;										// 修改值
}


12、得到表长

要做空表判断
结点从首元结点开始找,如果找到头结点就代表找完一遍了

int GetLength(CircList rear)
{
	if (IsEmptyList(rear))										// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	Position cur = rear->next->next;							// 标记首元结点
	int length = 0;												// 用来计数
	while (cur != rear->next)									// 找到头结点时代表找完一遍,结束循环
	{
		cur = cur->next;
		length++;
	}
	return length;
}

13、清空链表

要做空表判断
借用尾指针找到一个元素,用cur记录这个元素,尾指针指向下一个元素,再释放cur,直到尾指针重新指向头结点
最后使头结点指向头结点形成空表文章来源地址https://www.toymoban.com/news/detail-639391.html

void ClearList(CircList* ppRear)
{
	if (IsEmptyList(*ppRear))									// 空表则中断程序
	{
		puts("The list is empty.");
		exit(EXIT_FAILURE);
	}
	Position head = (*ppRear)->next;							// 记录头结点位置
	*ppRear = head->next;										// 将指针移动到首元结点
	// 记录指针指向的结点,让指针指向下一个结点,释放被记录的结点,直到指针指向头结点
	while ((*ppRear) != head)
	{
		Position cur = (*ppRear);
		*ppRear = cur->next;
		free(cur);
	}
	(*ppRear)->next = (*ppRear);								// 变回只有头结点的循环单链表
}

到了这里,关于带头结点和尾指针的循环单链表(C语言)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 上机实验二 设计单循环链表 西安石油大学数据结构

    (1)实验目的:掌握线性表的链式存储结构;掌握单循环链表及其基本操作的实现。 (2)主要内容:实现单循环链表的初始化、求数据元素个数、插入、删除、取数据元素等操作;用插入法建立带头结点的单循环链表;设计一个测试主函数验证所设计单循环链表的正确性。 掌握线性

    2024年02月07日
    浏览(61)
  • 【数据结构】C语言实现单链表(带头结点)

    Single linked list with leading nodes 关于不带头结点的单链表:不带头结点的单链表 结点定义: 接口定义: 初始化需要申请头结点,让头指针指向空的头结点。 将申请结点的代码进行封装: 需要越过头结点 找到尾结点,然后插入到尾结点的后面。 对比不带头结点的单链表的尾插

    2024年02月02日
    浏览(219)
  • 国内优秀的开源低代码框架:PagePlug,面向研发使用,拒绝重复、低价值的工单循环开发

    分享下Appsmith中文版的PagePlug吧, 开源、面向研发人员开发使用的低代码: PagePlug将开发人员的开发时间减少了 60%,PP框架本身解决了很多没必要的繁重工作。 前者appsmith目前是github上超29K最火的开源低代码平台,后者PagePlug也是目前国内开源社区比较火的低代码平台—— 针

    2024年02月09日
    浏览(38)
  • 【数据结构】C语言实现双向链表(带头结点、循环)

    结点定义: 接口定义: 我们将申请结点的代码封装成函数,方便后续使用 由于是带头结点的双向链表,因此在使用链表前,我们需要对链表进行初始化。 遍历链表,值得说的是,带头结点的双向链表的循环结束条件是 cur != phead 尾插时,需要先找到尾结点,然后将新结点插

    2024年02月03日
    浏览(72)
  • 数据结构(2)—单链表(带头结点和不带头结点)

            单链表 是通过一组任意的存储单元来存储线性表中的数据元素。每个结点都有 data数据域 (用来存放数据元素)和 next指针域 (用来存放后继节点的地址)。         对于顺序表,单链表可以解决顺序表需要一整个大量的连续的存储单元的缺点,单链表的元素

    2024年02月05日
    浏览(56)
  • 【数据结构与算法分析】使用C语言实现队列的两种(带头结点与不带头结点)链式存储,并且给出一种循环队列的设计思想

      当我们编写程序时,经常需要处理各种数据结构。队列是一种常见的数据结构,它有着广泛的应用场景。队列的基本操作包括入队和出队,应用于模拟等待队列、消息队列、计算机缓存等场合。   在实际编程中,我们可以用不同的数据结构来实现队列。本文主要介绍了

    2024年02月08日
    浏览(118)
  • 王道p40 17.设计一个算法用于判断带头结点的循环双链表是否对称(c语言代码实现)

    补充循环双链表的知识: 循环双链表是一种链表数据结构,在链表的基础上增加了头尾相连的循环特性,即链表的最后一个节点指向第一个节点,同时每个节点除了储存下一个节点的指针外还储存前一个节点的指针,这样可以实现在链表两端快速插入和删除元素的操作。 与

    2024年02月07日
    浏览(46)
  • 带头结点单链表【详细解析+完整代码】

    它是通过一组任意的储存单元来存储线性表中的数据元素。为建立线性关系,每个结点需要一个指针域以及指向下一结点的指针域。带头结点链表头节点不存储数据。 结点结构: 带头结点链表结构: 保证了每个结点都有前驱结点,因此在链表上第一个结点的操作与其他结点操

    2024年04月15日
    浏览(43)
  • 单链表创建之--头插法创建带头结点的单链表

    单链表常见的创建方法有 头插法 和 尾插法 ,这里记录头插法创建 带头结点的单链表 具体过程: 以C语言为例, 1)首先使用  typedef  定义结点数据类型 4行的 LNode 和 * LinkList 可有可无,有的话后面定义结点变量和指针变量时更方便,不必须在LNode前面加 struct

    2024年02月06日
    浏览(39)
  • 【数据结构】——单链表的基本操作(带头结点)

            单链表解决了顺序表需要大量连续存储单元的缺点,但单链表附加指针域, 存储密度较顺序表低(考点!!) 。由于单链表的元素离散地分布在存储空间中,所以单链表是 非随机存取 的存储结构,即不能直接找到表中某个特定的结点。当查找某个特定结点时,需要

    2024年02月05日
    浏览(52)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包