【数据结构与算法】 - 双向链表 - 详细实现思路及代码

这篇具有很好参考价值的文章主要介绍了【数据结构与算法】 - 双向链表 - 详细实现思路及代码。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


【数据结构与算法】 - 双向链表 - 详细实现思路及代码

一、概述

前几篇文章介绍了怎样去实现单链表、单循环链表,这篇文章主要介绍双向链表以及实现双向链表的步骤,最后提供我自己根据理解实现双向链表的C语言代码。跟着后面实现思路看下去,应该可以看懂代码,看懂代码后,就对双向链表有了比较抽象的理解了,最后自己再动手写一个双向链表,就基本理解这个东西了。
【数据结构与算法】 - 双向链表 - 详细实现思路及代码

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

二、双向链表

双向链表:在单链表的每个结点中,再设置一个指向其前驱结点的指针域。
下图是 单链表
【数据结构与算法】 - 双向链表 - 详细实现思路及代码

下图是 双向链表
【数据结构与算法】 - 双向链表 - 详细实现思路及代码

双向链表的特点:

  1. 双向链表可以反向访问到链表的结点,因为它有指向前一个结点的指针prior
  2. 带有头结点的双向链表,为空链表时,头结点的两个指针域都指向NULL
    【数据结构与算法】 - 双向链表 - 详细实现思路及代码
  3. 带有头结点的双向链表,为非空链表时,
    头结点的前驱指针域指向NULL,后驱指针域指向第一个结点;
    最后一个结点的前驱指针域指向前一个结点,后驱指针域指向NULL
    其他结点的前驱指针域指向前一个结点,后驱指针域指向后一个结点;
    【数据结构与算法】 - 双向链表 - 详细实现思路及代码

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

三、双向链表实现步骤

从上面知道了双向链表的相关概念和一些特点,接下来开始实现双向链表,这里使用带有头结点的双向链表进行讲解,从初始化双向链表、插入数据、删除数据、查找数据、销毁双向链表5个操作进行说明,需要注意的是,双向链表的插入、删除操作需要改变两个指针域;其他操作基本和单链表一致。

📌3.1 C语言定义双向链表结点

为了和前几篇文章的链表做比较,双向链表结构体也尽量定义相似的。

typedef int ElemType;
typedef struct _DoubleListNode
{
	ElemType data;
	struct _DoubleListNode *prior;	// 前驱指针
	struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;

📌3.2 双向链表初始化

因为带有头结点,初始化时就需要分配一个头结点的内存空间,且头指针会一直指向头结点。
双向链表初始化算法思路如下:

1、分配一个结点的存储空间作为头结点,并将头指针指向头结点;
2、让头结点的 prior指针 和 next指针 都指向NULL,头结点的数据填一个无效值;
3、将头指针返回给函数调用者。

C语言实现代码如下:

DoubleLinkList ListInit()
{
	DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));
	list->prior = NULL;
	list->next = NULL;
	list->data = -1;
	return list;
}

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

📌3.3 双向链表插入数据

双向链表插入数据大致分为两个步骤:首先,找到插入位置n的前一个结点;其次,是插入新结点,可以:先连接新结点、再指向新结点的顺序。
先连接新结点:是先把新结点的两个指针域分别连接当前结点和下个结点,new->prior = cur;new->next = cur->next;
再指向新结点:将当前节点的的指针域指向新节点,与旧节点断开,cur->next->prior = new;cur->next = new;
【数据结构与算法】 - 双向链表 - 详细实现思路及代码

双向链表在第n个位置插入数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是插入位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是当前结点无效,说明链表长度不够;
5、否则,说明当前结点cur的下个位置就是插入位置n,分配存储空间给新结点new;
6、把值填进新节点的数据域,用新结点prior指向当前结点,next指向当前节点的下个节点;
7、再将下个结点的prior指向新结点,当前结点的next指向新结点,完成插入操作。

C语言实现代码如下:

int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{
	if(list==NULL || n<1) // 判断参数有效性
		return -1;
		
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
		int cur_i=0;			// cur_i表示当前结点的序号,0-头结点
	while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));
	new->data = data;
	new->prior = cur;
	new->next = cur->next;
	if(cur->next)		// 在最后一个结点插入时,cur->next==NULL
		cur->next->prior = new;
	cur->next = new;
	
	return 0;
}

📌3.4 双向链表删除数据

双向链表删除结点也是需要改变两个指针域,大致步骤如下,首先,找到删除位置n的前一个结点;其次,“把前一个结点的next指针域指向删除结点del的下个结点”,“再把下个结点的prior指针域指向删除结点del的前个结点”,这样就删除了下一个结点。
【数据结构与算法】 - 双向链表 - 详细实现思路及代码

双向链表删除第n个数据的算法思路:

1、定义一个结点指针cur指向头结点,用来遍历链表;
2、定义一个变量cur_i,用来表示下个结点的序号,初始化为0表示当前指向头结点;
3、将cur指针不断往后移动,直到下个位置就是删除位置n,即当cur_i==(n-1)跳出循环;
4、若结束循环后是最后一个结点(cur->next==NULL),说明链表长度不够;
5、否则,说明下个结点(cur->next)就是删除位置n的结点delete,赋值delete = cur->next;
6、将前一个结点的next指针域指向 del 的下个结点 ,delete->prior->next = delete->next;
7、将下一个结点的prior指针域指向 del 的前个结点 ,delete->next->prior = delete->prior;;
8、最后释放delete结点的内存,完成删除操作。

C语言实现代码如下,删除结点更关注的是下个结点(cur->next)的有效性:

// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next && cur_i<(n-1))
	{// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(!cur->next)		// 下个结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;		// 链表没有 n 那么长
	}
	DoubleListNode *delete = cur->next;
	delete->prior->next = delete->next;
	delete->next->prior = delete->prior;
	free(delete);
	return 0;
}

📌3.5 双向链表查找数据

查找数据时,将指针指向第一个结点而非头结点,下面函数中list是头指针,指向头结点,双向链表非空时,list->next就是第一个结点;双向链表为空时,list->next == NULL。双向链表 和 单链表 查找数据的算法是一样的。

双向链表查找第n个数据的算法思路:

1、定义一个结点指针cur指向第一个结点(list->next),用来遍历链表;
2、定义一个变量cur_i,用来表示当前结点的序号,初始化为1(第一步指向的就是第一个结点);
3、若当前结点有效,且当前位置不是查找位置n,就继续后移,直到最后结点或cur_i==n跳出循环;
4、若结束循环后,当前结点无效,说明已经移动到最后,链表长度不够;
5、否则,说明当前结点(cur)就是查找位置n的结点;返回结点数据*data = cur->data。

C语言实现代码如下:

int ListFind(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	DoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// i表示当前结点的序号
	while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	*data = cur->data;
	printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);
	return 0;
}

📌3.6 双向链表的销毁

双向链表销毁的算法思路:

1、定义一个结点指针cur指向第一个结点,用来遍历链表;
2、定义一个结点指针next,保存下个结点地址;
3、当前指针不是指向最后一个结点的指针域就后移,进入循环:
	3.1、先保存下个结点地址,因为下个结点本来保存在cur->next,直接free(cur)会丢掉下个结点;
	3.2、删除当前结点,释放内存
	3.3、将当前指针指向前面保存好的下个结点。
4、结束循环后,已经删除完所有节点,此时需要将头结点的两个指针域都指向NULL,表示空链表。

C语言实现代码如下:

void ListDestroy(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;	// 指向第一个节点
	DoubleListNode* next = NULL;		// 用于保存下个结点地址
	while(cur)	// 当前结点有效,就往后移动
	{
		next = cur->next;		// 保存下个结点地址
		//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);
		free(cur);				// 删除当前结点、并释放内存
		cur = next;				// 将当前结点指针指向下个结点
	}
	list->prior = NULL;
	list->next = NULL;
}

【数据结构与算法】 - 双向链表 - 详细实现思路及代码

四、双向链表完整代码

代码只是为了更好地了解循环链表,实现过程可能存在不足,有发现的,欢迎指正,谢谢!!!
代码已在Ubuntu编译通过,可执行。文章来源地址https://www.toymoban.com/news/detail-428261.html

// DoubleList.c
#include <stdio.h>
#include <stdlib.h>

typedef int ElemType;
typedef struct _DoubleListNode
{
	ElemType data;
	struct _DoubleListNode *prior;	// 前驱指针
	struct _DoubleListNode *next;	// 后驱指针
}DoubleListNode;
typedef DoubleListNode* DoubleLinkList;

DoubleLinkList ListInit()
{
	DoubleLinkList list = (DoubleLinkList)malloc(sizeof(DoubleListNode));
	list->prior = NULL;
	list->next = NULL;
	list->data = -1;
	return list;
}

int ListInsert(DoubleLinkList list, int data, int n)// 将node插入到第n位,n从1开始
{
	if(list==NULL || n<1) // 判断参数有效性
		return -1;
		
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur && cur_i<(n-1))// 当前结点有效,且不是插入位置的前一个结点,就后移一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	DoubleListNode* new = (DoubleListNode*)malloc(sizeof(DoubleListNode));
	new->data = data;
	new->prior = cur;
	new->next = cur->next;
	if(cur->next)		// 在最后一个结点插入时,cur->next==NULL
		cur->next->prior = new;
	cur->next = new;
	
	return 0;
}

// 删除第n个结点,且将删除的值通过data传出
int ListDelete(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	DoubleListNode* cur = list;	// cur指向当前结点,初始化指向头结点
	int cur_i=0;				// cur_i表示当前结点的序号,0-头结点
	while(cur->next && cur_i<(n-1))
	{// 下个结点有效,且当前位置不是删除位置的前一个,就后移一个
		cur = cur->next;
		cur_i++;
	}
	if(!cur->next)		// 下个结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;		// 链表没有 n 那么长
	}
	DoubleListNode *delete = cur->next;
	delete->prior->next = delete->next;
	delete->next->prior = delete->prior;
	free(delete);
	return 0;
}

int ListFind(DoubleLinkList list, int *data, int n)
{
	if(list==NULL || data==NULL || n<1)
		return -1;
	
	DoubleListNode* cur = list->next;// 指向第一个节点
	int cur_i=1;			// i表示当前结点的序号
	while(cur && cur_i<n)	// 当前结点有效,且当前位置不是查找位置n,就往后移动一个
	{
		cur = cur->next;
		cur_i++;
	}
	if(!cur)			// 当前结点无效,说明已经移动到最后
	{
		printf("[%s %d]error din't have No.%d\n", __FUNCTION__,__LINE__, n);
		return -1;	// 链表没有 n 那么长
	}
	*data = cur->data;
	printf("[%s %d]find No.%d = %d\n", __FUNCTION__,__LINE__, n,*data);
	return 0;
}

void ListDestroy(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;	// 指向第一个节点
	DoubleListNode* next = NULL;		// 用于保存下个结点地址
	while(cur)	// 当前结点有效,就往后移动
	{
		next = cur->next;		// 保存下个结点地址
		//printf("[%s %d]delete %d\n", __FUNCTION__,__LINE__, cur->data);
		free(cur);				// 删除当前结点、并释放内存
		cur = next;				// 将当前结点指针指向下个结点
	}
	list->prior = NULL;
	list->next = NULL;
}

void ListPrintf(DoubleLinkList list)
{
	DoubleListNode* cur = list->next;// 指向第一个节点
	printf("list:[");
	while(cur)
	{
		printf("%d,",cur->data);
		cur = cur->next;
	}
	printf("]\n");
}

int main()
{
	DoubleLinkList list=ListInit();
	int data=0;
	
	printf("Linklist is empty !!! \n");
	ListInsert(list, 2, 2);		// 空链表时,验证插入
	ListDelete(list, &data, 1);	// 空链表时,验证删除
	ListFind(list, &data, 1);	// 空链表时,验证查询
	ListDestroy(list);			// 空链表时,验证销毁
	
	printf("\ninsert 3 data\n");
	// 正常插入3个数据
	ListInsert(list, 1, 1);
	ListInsert(list, 2, 2);
	ListInsert(list, 3, 3);
	ListPrintf(list);
	
	printf("\n验证错误值\n");
	ListInsert(list, 5, 5);		// 验证插入
	ListDelete(list, &data, 4);	// 验证删除
	ListFind(list, &data, 4);	// 验证查询
	
	printf("\n正常操作\n");
	// 正常操作
	ListFind(list, &data, 2);
	printf("delete 2,now\n");
	ListDelete(list, &data, 2);
	ListPrintf(list);
	
	printf("Insert 4 to 2,now\n");
	ListInsert(list, 4, 2);
	ListPrintf(list);
	
	printf("Destroy ,now\n");
	ListDestroy(list);
	ListPrintf(list);

	return 0;
}

到了这里,关于【数据结构与算法】 - 双向链表 - 详细实现思路及代码的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 【数据结构和算法】实现带头双向循环链表(最复杂的链表)

    【数据结构和算法】实现带头双向循环链表(最复杂的链表)

    前文,我们实现了认识了链表这一结构,并实现了无头单向非循环链表,接下来我们实现另一种常用的链表结构,带头双向循环链表。如有仍不了解单向链表的,请看这一篇文章(7条消息) 【数据结构和算法】认识线性表中的链表,并实现单向链表_小王学代码的博客-CSDN博客

    2024年01月17日
    浏览(14)
  • 数据结构与算法:双向链表

    数据结构与算法:双向链表

    朋友们大家好啊,在上节完成单链表的讲解后,我们本篇文章来对 带头循环双向链表进行讲解 单链表中,一个节点存储数据和指向下一个节点的指针,而双向链表除了上述两个内容,还包括了 指向上一个节点的指针 带头的双向链表,是指在双向链表的最前端添加了一个 额

    2024年02月20日
    浏览(13)
  • 【数据结构与算法】双向链表

    【数据结构与算法】双向链表

    作者:旧梦拾遗186 专栏:数据结构成长日记   带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。 现在我们来通

    2024年02月11日
    浏览(18)
  • 【数据结构篇】手写双向链表、单向链表(超详细)

    【数据结构篇】手写双向链表、单向链表(超详细)

    什么是链表 ? 链表(Linked List)是用链式存储结构实现的线性表。链表示意图: 链表的组成 : 数据域 + 引用域 (数据域和引用域合称结点或元素) 数据域存放数据元素自身的数据 引用域存放相邻结点的地址 链表的特点 : 链表中元素的联系依靠引用域 具有线性结构的特

    2024年02月11日
    浏览(18)
  • 数据结构与算法(四):双向链表

    数据结构与算法(四):双向链表

    双向链表概念和单向链表是一致的,区别在于双向链表在单向链表的基础上,指针区域多了一个指向上一个节点的指针。单向链表内容可以参考我的上一篇文章:http://t.csdn.cn/Iu56H。 基本的数据结构如图所示: 双向链表结构包含了节点的数据内容和两个指针:指向前一个节点

    2024年02月14日
    浏览(15)
  • 数据结构-双向链表(c++)超全超详细

    数据结构-双向链表(c++)超全超详细

    单链表结点中只有一个指向其后继的指针,使得单链表只能从头结点依次顺序地向后遍历。要访问某个结点的前驱结点(插入,删除操作时),只能从头开始遍历,访问后继结点的时间复杂度为O(1),访问前驱结点的时间复杂度为O(n)。 提示:以下是本篇文章正文内容,下面案

    2023年04月08日
    浏览(14)
  • 数据结构——实现双向链表

    数据结构——实现双向链表

    怎么说呢?光乍一听名字好像很难的样子是吧,那如果你这样认为的话,可就要让你大跌眼镜了哦,其实双向带头循环链表从操作和理解上来说都是要易于单项不带头不循环链表(俗称单链表)的。 咱们就来见识见识吧!希望真的能让你们“大跌眼镜”哈! 双向带头循环链

    2024年02月07日
    浏览(12)
  • 数据结构:详解【链表】的实现(单向链表+双向链表)

    数据结构:详解【链表】的实现(单向链表+双向链表)

    1.顺序表的问题和思考 问题: 中间/头部的插入删除,时间复杂度为O(N)。 增容需要申请新空间,拷贝数据,释放旧空间,会有不小的消耗。 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再继续插入了5个数据,后面没有数据

    2024年03月26日
    浏览(12)
  • 【(数据结构)— 双向链表的实现】

    【(数据结构)— 双向链表的实现】

    注意: 这里的 “带头” 跟前面我们说的 “头节点” 是两个概念,实际前面的在单链表阶段称呼不严 谨,但是为了同学们更好的理解就直接称为单链表的头节点。 带头链表里的头节点,实际为 “哨兵位” ,哨兵位节点不存储任何有效元素,只是站在这里“放哨 的” “哨

    2024年02月06日
    浏览(14)
  • 【数据结构】双向链表的实现

    【数据结构】双向链表的实现

    我要扼住命运的咽喉,他却不能使我完全屈服。                      --贝多芬 目录 一.带头循环的双向链表的特点 二.不带头不循环单向链表和带头循环的双向链表的对比 三.初始化链表,创建哨兵结点 四.双向链表的各种功能的实现 1.双向链表的尾插 2.双向链表的打印 

    2023年04月10日
    浏览(9)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包