/* c2-4.h 线性表的双向链表存储结构 */ typedef struct DuLNode { ElemType data; struct DuLNode *prior,*next; }DuLNode,*DuLinkList;
/* bo2-5.c 双链循环线性表(存储结构由c2-4.h定义)的基本操作(14个) */ Status InitList(DuLinkList *L) { /* 产生空的双向循环链表L */ *L=(DuLinkList)malloc(sizeof(DuLNode)); if(*L) { (*L)->next=(*L)->prior=*L; return OK; } else return OVERFLOW; } Status DestroyList(DuLinkList *L) { /* 操作结果:销毁双向循环链表L */ DuLinkList q,p=(*L)->next; /* p指向第一个结点 */ while(p!=*L) /* p没到表头 */ { q=p->next; free(p); p=q; } free(*L); *L=NULL; return OK; } Status ClearList(DuLinkList L) /* 不改变L */ { /* 初始条件:L已存在。操作结果:将L重置为空表 */ DuLinkList q,p=L->next; /* p指向第一个结点 */ while(p!=L) /* p没到表头 */ { q=p->next; free(p); p=q; } L->next=L->prior=L; /* 头结点的两个指针域均指向自身 */ return OK; } Status ListEmpty(DuLinkList L) { /* 初始条件:线性表L已存在。操作结果:若L为空表,则返回TRUE,否则返回FALSE */ if(L->next==L&&L->prior==L) return TRUE; else return FALSE; } int ListLength(DuLinkList L) { /* 初始条件:L已存在。操作结果:返回L中数据元素个数 */ int i=0; DuLinkList p=L->next; /* p指向第一个结点 */ while(p!=L) /* p没到表头 */ { i++; p=p->next; } return i; } Status GetElem(DuLinkList L,int i,ElemType *e) { /* 当第i个元素存在时,其值赋给e并返回OK,否则返回ERROR */ int j=1; /* j为计数器 */ DuLinkList p=L->next; /* p指向第一个结点 */ while(p!=L&&j<i) /* 顺指针向后查找,直到p指向第i个元素或p指向头结点 */ { p=p->next; j++; } if(p==L||j>i) /* 第i个元素不存在 */ return ERROR; *e=p->data; /* 取第i个元素 */ return OK; } int LocateElem(DuLinkList L,ElemType e,Status(*compare)(ElemType,ElemType)) { /* 初始条件:L已存在,compare()是数据元素判定函数 */ /* 操作结果:返回L中第1个与e满足关系compare()的数据元素的位序。 */ /* 若这样的数据元素不存在,则返回值为0 */ int i=0; DuLinkList p=L->next; /* p指向第1个元素 */ while(p!=L) { i++; if(compare(p->data,e)) /* 找到这样的数据元素 */ return i; p=p->next; } return 0; } Status PriorElem(DuLinkList L,ElemType cur_e,ElemType *pre_e) { /* 操作结果:若cur_e是L的数据元素,且不是第一个,则用pre_e返回它的前驱, */ /* 否则操作失败,pre_e无定义 */ DuLinkList p=L->next->next; /* p指向第2个元素 */ while(p!=L) /* p没到表头 */ { if(p->data==cur_e) { *pre_e=p->prior->data; return TRUE; } p=p->next; } return FALSE; } Status NextElem(DuLinkList L,ElemType cur_e,ElemType *next_e) { /* 操作结果:若cur_e是L的数据元素,且不是最后一个,则用next_e返回它的后继, */ /* 否则操作失败,next_e无定义 */ DuLinkList p=L->next->next; /* p指向第2个元素 */ while(p!=L) /* p没到表头 */ { if(p->prior->data==cur_e) { *next_e=p->data; return TRUE; } p=p->next; } return FALSE; } DuLinkList GetElemP(DuLinkList L,int i) /* 另加 */ { /* 在双向链表L中返回第i个元素的位置指针(算法2.18、2.19要调用的函数) */ int j; DuLinkList p=L; for(j=1;j<=i;j++) p=p->next; return p; } Status ListInsert(DuLinkList L,int i,ElemType e) /* 改进算法2.18 */ { /* 在带头结点的双链循环线性表L中第i个位置之前插入元素e,i的合法值为1≤i≤表长+1 */ DuLinkList p,s; if(i<1||i>ListLength(L)+1) /* i值不合法 */ return ERROR; p=GetElemP(L,i-1); /* 在L中确定第i-1个元素的位置指针p */ if(!p) /* p=NULL,即第i-1个元素不存在 */ return ERROR; s=(DuLinkList)malloc(sizeof(DuLNode)); if(!s) return OVERFLOW; s->data=e; /* 在第i-1个元素之后插入 */ s->prior=p; s->next=p->next; p->next->prior=s; p->next=s; return OK; } Status ListDelete(DuLinkList L,int i,ElemType *e) /* 算法2.19 */ { /* 删除带头结点的双链循环线性表L的第i个元素,i的合法值为1≤i≤表长+1 */ DuLinkList p; if(i<1||i>ListLength(L)) /* i值不合法 */ return ERROR; p=GetElemP(L,i); /* 在L中确定第i个元素的位置指针p */ if(!p) /* p=NULL,即第i个元素不存在 */ return ERROR; *e=p->data; p->prior->next=p->next; p->next->prior=p->prior; free(p); return OK; } void ListTraverse(DuLinkList L,void(*visit)(ElemType)) { /* 由双链循环线性表L的头结点出发,正序对每个数据元素调用函数visit() */ DuLinkList p=L->next; /* p指向头结点 */ while(p!=L) { visit(p->data); p=p->next; } printf("\n"); } void ListTraverseBack(DuLinkList L,void(*visit)(ElemType)) { /* 由双链循环线性表L的头结点出发,逆序对每个数据元素调用函数visit()。另加 */ DuLinkList p=L->prior; /* p指向尾结点 */ while(p!=L) { visit(p->data); p=p->prior; } printf("\n"); }
相关推荐
Linux操作系统中通用双向循环链表的实现分析 Linux操作系统是一个支持多用户、多任务、多线程和多CPU的开源操作系统,其内核功能强大、性能稳定并具有丰富的应用软件支持。Linux内核源代码主要由C语言和少量的汇编...
本文将深入探讨如何利用单向链表和双向循环链表来构建这样一个系统。我们将讨论标题中的"航班订票系统"、"单向链表"和"双向循环链表"这些核心概念。 首先,我们来看"单向链表"。单向链表是一种线性数据结构,每个...
### 数据结构课程设计报告:基于双向循环链表的通讯录设计 #### 概要设计 在沈阳航空航天大学的计算机学院,计算机科学与技术专业的学生冯读庆在其数据结构课程设计中,选择了一个既具挑战性又实用的项目——基于...
在本实验中,我们利用双向循环链表来实现长整数的存储和四则运算,特别是加法和减法。这种数据结构的选择主要是因为它能够方便地处理长整数的存储和运算过程中的进位和借位操作。 首先,双向循环链表的每个节点仅...
双向循环链表是一种高级的数据结构,常用于需要前后移动指针的场景,如实现LRU缓存淘汰策略、编辑器的撤销重做功能等。本项目以C++语言实现了带头结点的双向循环链表,这将有助于我们深入理解这一概念。 首先,双向...
双向循环链表是一种重要的数据结构,它在计算机科学和编程中有着广泛的应用。与单向链表不同,双向循环链表允许元素在正向和反向两个方向上进行遍历,这种特性使得它在需要频繁地进行前向或后向移动操作时特别有用。...
双向循环链表是一种特殊的数据结构,它扩展了单链表的概念,允许在链表的任一位置进行前向和后向遍历。在这个话题中,我们将深入探讨如何使用C++来实现双向循环链表,并涵盖其主要功能,如定点插入、定点删除以及...
双向循环链表是一种特殊的数据结构,它允许我们从两个方向遍历元素。与单向链表不同,双向循环链表的每个节点都有两个指针,一个指向其前一个节点,另一个指向其后一个节点。这种结构在实现队列、栈和其他高级数据...
本文将深入探讨双向循环链表的原理、实现方式以及其在实际应用中的价值。 双向循环链表是一种特殊的链表类型,与普通的单向链表不同,它具有两个指针,一个指向前一个节点,另一个指向后一个节点。这种设计使得在...
### 双向循环链表在 LINUX Kernel 中的实现 #### 一、引言 双向循环链表作为一种常用的数据结构,在 Linux 内核中扮演着重要角色。与传统的链表实现不同,Linux 内核中的双向循环链表采用了更加高效且灵活的设计...
本文将深入探讨“简单的双向循环链表”,并结合提供的`LinkedList.java`源码来理解其实现原理。 双向循环链表与单向链表不同,它在每个节点中不仅保存了指向下一个节点的指针,还保存了指向前一个节点的指针,这种...
【双向循环链表】是一种特殊的数据结构,它允许节点在链表中前后两个方向移动。在华科计算机学院的数据结构实验中,学生们被要求实现一个使用C++编程的双向循环链表,包括一系列操作,如创建、销毁、置空、求长度、...
【带头结点的双向循环链表数据结构】 在数据结构中,双向循环链表是一种特殊类型的数据结构,它允许从两个方向遍历链表。在本案例中,我们需要使用C++和Java分别实现这种数据结构,并确保它们符合指定的要求。 在...
本文将详细讨论如何使用C++实现一个基于双向循环链表的派生栈和队列。 首先,我们要理解双向循环链表的基本概念。双向循环链表是一种链式存储结构,每个节点包含数据和两个指针,分别指向其前一个节点和后一个节点...
**C++ 双向循环链表的基本操作** 在C++编程中,数据结构是至关重要的一个部分,其中链表作为一种动态数据结构,被广泛应用于各种算法和程序设计中。双向循环链表是一种特殊的链表类型,它允许在链表中的元素之间...
《双向循环链表的C++实现详解》 双向循环链表是一种高级的数据结构,它不仅包含了一般单向链表的特性,还允许从链表的任一节点出发,既能向前遍历也能向后遍历。在C++中实现双向循环链表,需要对链表的节点结构和...
单链表实现双向循环链表单向链表存在一个弊端就是,当需要获取某个结点p的前驱时,需要从头指针开始遍历链表,获得“前驱”的执行时间为O(n),为了克服单向链表的这种缺点,可以利用双向链表。在双向链表中有两个...
Linux内核中的双向循环链表是一种高效且灵活的数据结构,被广泛应用于内核的各种数据管理中,如进程、文件、模块等。这种链表的设计允许快速地添加、删除和遍历元素,而且具有面向对象的特性,使得它可以适应不同的...