IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 【数据结构】必会的两种链表结构之---双向带头循环链表(附源码) -> 正文阅读

[数据结构与算法]【数据结构】必会的两种链表结构之---双向带头循环链表(附源码)

目录

前言

双向带头循环链表

结构

实现

函数声明

打印链表

初始化链表

销毁链表

申请结点

尾插

头插

尾删

头删

查找

在pos位置前面插入

删除pos位置的结点


前言

前面我们讲过,链表分为带头和不带头,双向和单向,循环和非循环链表。

那么经过排列组合之后,就会有8中链表结构。

其中最常用的是单向不带头非循环链表和双向带头循环链表。

前者在之前已经实现过了,今天的内容主要是双向带头循环链表的实现。

双向带头循环链表

结构

1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多

2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了,后面我们代码实现了就知道了。
?

实现

函数声明

注意:

  1. 因为是双向链表所以在结构体中不仅会保存下一个结点的地址,而且会保存前一个结点的地址。
  2. 使用哨兵位的头结点就不会改变头结点,所以在传参时就不需要传二级指针。
#pragma once


#include<stdio.h>
#include<stdlib.h>
#include<assert.h>

typedef int DataType;

typedef struct ListNode
{
	DataType data;
	struct ListNode* next;
	struct ListNode* prev;
}ListNode;


//打印链表
void PrintList(ListNode* phead);

//初始化
ListNode* Init(ListNode* phead);

//销毁链表
void ListDestroy(ListNode* pphead);

//申请结点
ListNode* BuyNode(DataType x);

//尾插
void ListPushBack(ListNode* phead, DataType x);

//头插
void ListPushFront(ListNode* phead, DataType x);


//尾删
void ListPopBack(ListNode* phead);

//头删
void ListPopFront(ListNode* phead);

//查找
ListNode* ListFind(ListNode* phead, DataType x);

// 在pos的前面进行插入
void ListInsert(ListNode* pos, DataType x);

// 删除pos位置的节点
void ListErase(ListNode* pos);

打印链表

这里与单链表的区别在于,如何判断打印结束。

使用cur指针遍历链表,初始化 cur = phead->next 循环链表中尾结点不会指向NULL,而是会指向头结点。

那么当 cur = phead 时循环结束,打印完成。

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

初始化链表

初始化时需先创建头结点,但这样会改变头结点,为了使原结构体指针改变,这里可以传二级指针或者返回头结点。但后面的参数我们都会传一级指针,为了使代码具有一致性,这里我们选择用返回值的方法改变原指针。

ListNode* Init(ListNode* phead)
{
	
	phead = (ListNode*)malloc(sizeof(ListNode));
	if (phead == NULL)
	{
		perror("malloc");
		exit(-1);
	}
	phead->next = phead;//前后指针均指向自己形成循环
	phead->prev = phead;
	return phead;
}

销毁链表

同单链表一样,使用循环依次释放。

但不要忘记释放头结点!

void ListDestroy(ListNode* phead)
{
	assert(phead);
	ListNode* cur = phead->next;
	while (cur!=phead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(phead);//释放头结点
	phead = NULL;
}

申请结点

ListNode* BuyNode(DataType x)
{
	ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));
	if (newnode == NULL)
	{
		perror("malloc");
		exit(-1);
	}
	newnode->data = x;
	newnode->next = newnode;
	newnode->prev = newnode;
	return newnode;
}

尾插

尾插相对于单链表来说更加简单,因为这里不需找尾结点。

phead->prev就是尾结点,而且只有头结点时也能解决。

时间复杂度为O(1)。

void ListPushBack(ListNode* phead, DataType x)
{
	assert(phead);
	ListNode* newnode = BuyNode(x);//申请结点
	ListNode* tail = phead->prev;//保存尾结点
	tail->next = newnode;//尾插
	phead->prev = newnode;
	newnode->prev = tail;
	newnode->next = phead;
}

头插

保存第一个有效结点的地址,将新结点插入。

与单链表不同的是,这里不需要移动头结点。

void ListPushFront(ListNode* phead, DataType x)
{
	assert(phead);
	ListNode* newnode = BuyNode(x);
	ListNode* next = phead->next;
	phead->next = newnode;
	newnode->next = next;
	newnode->prev = phead;
	next->prev = newnode;
}

尾删

找到尾结点的前一个位置的地址,将他的next指向结点,而头结点的prev指向他。

再释放掉原尾结点。

注意:删除结点时需保证链表中存在有效结点。

void ListPopBack(ListNode* phead)
{
	assert(phead && phead->next != phead);
	ListNode* tail = phead->prev;
	ListNode* tailprev = tail->prev;
	tailprev->next = phead;
	phead->prev = tailprev;
	free(tail);
	tail = NULL;
}

头删

与尾删类似,保存第一个有效节点的下一个结点的地址,

将头结点的next指向他,他的prev指向头结点,

最后释放掉原第一个有效结点。

void ListPopFront(ListNode* phead)
{
	assert(phead && phead->next != phead);
	ListNode* next = phead->next;
	ListNode* nextnext = next->next;
	phead->next = nextnext;
	nextnext->prev = phead;
	free(next);
	next = NULL;
}

查找

ListNode* ListFind(ListNode* phead, DataType x)
{
	assert(phead);
	ListNode* cur = phead->next;
	while (cur != phead)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

在pos位置前面插入

与单链表不同,单链表在pos前面插入时,会比较复杂,需要找到pos位置前面的地址。

而双向链表中,在pos前面和后面插入均可。时间复杂度都是O(1)。

而且这个函数相当于可以实现任何位置的插入,当然前面的尾插,头插都可复用此函数。

void ListInsert(ListNode* pos, DataType x)
{
	assert(pos);
	ListNode* prev = pos->prev;
	ListNode* newnode = BuyNode(x);
	prev->next = newnode;
	newnode->prev = prev;
	newnode->next = pos;
	pos->prev = newnode;
}

删除pos位置的结点

同样此函数也可以删除任意位置的结点(头结点除外),

前面的头删,尾删也可以复用此函数。

void ListErase(ListNode* pos)
{
	assert(pos);
	ListNode* prev = pos->prev;
	ListNode* next = pos->next;
	prev->next = next;
	next->prev = prev;
	free(pos);
	pos = NULL;
}

?

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-11-02 23:15:34  更:2021-11-02 23:15:36 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/8 4:19:31-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码