02循环单链表

Posted 蒋酱酱

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了02循环单链表相关的知识,希望对你有一定的参考价值。

循环单链表定义:将单链表中终端结点的指针端由空指针改为指向头结点,就使整个单链表形成了

        一个环,这种头尾相接的单链表成为单循环链表。

循环链表的数据结构:

1 /* c2-2.h 线性表的单链表存储结构 */
2 struct LNode
3 {
4     ElemType data;
5     struct LNode *next;
6 };
7 typedef struct LNode *LinkList; /* 另一种定义LinkList的方法 */

代码实现:

  1  
  2 
  3 /* bo2-4.c 设立尾指针的单循环链表(存储结构由c2-2.h定义)的12个基本操作 */
  4  Status InitList_CL(LinkList *L)
  5  { 
  6      /* 操作结果:构造一个空的线性表L */
  7    *L=(LinkList)malloc(sizeof(struct LNode)); /* 产生头结点,并使L指向此头结点 */
  8    if(!*L) /* 存储分配失败 */
  9      exit(OVERFLOW);
 10    (*L)->next=*L; /* 指针域指向头结点 */
 11    return OK;
 12  }
 13 
 14  Status DestroyList_CL(LinkList *L)
 15  { 
 16      /* 操作结果:销毁线性表L */
 17    LinkList q,p=(*L)->next; /* p指向头结点 */
 18    while(p!=*L) /* 没到表尾 */
 19    {
 20      q=p->next;
 21      free(p);
 22      p=q;
 23    }
 24    free(*L);
 25    *L=NULL;
 26    return OK;
 27  }
 28 
 29  Status ClearList_CL(LinkList *L) /* 改变L */
 30  {
 31      /* 初始条件:线性表L已存在。操作结果:将L重置为空表 */
 32    LinkList p,q;
 33    *L=(*L)->next; /* L指向头结点 */
 34    p=(*L)->next; /* p指向第一个结点 */
 35    while(p!=*L) /* 没到表尾 */
 36    {
 37      q=p->next;
 38      free(p);
 39      p=q;
 40    }
 41    (*L)->next=*L; /* 头结点指针域指向自身 */
 42    return OK;
 43  }
 44 
 45  Status ListEmpty_CL(LinkList L)
 46  { 
 47      /* 初始条件:线性表L已存在。操作结果:若L为空表,则返回TRUE,否则返回FALSE */
 48    if(L->next==L) /**/
 49      return TRUE;
 50    else
 51      return FALSE;
 52  }
 53 
 54  int ListLength_CL(LinkList L)
 55  {
 56      /* 初始条件:L已存在。操作结果:返回L中数据元素个数 */
 57    int i=0;
 58    LinkList p=L->next; /* p指向头结点 */
 59    while(p!=L) /* 没到表尾 */
 60    {
 61      i++;
 62      p=p->next;
 63    }
 64    return i;
 65  }
 66 
 67  Status GetElem_CL(LinkList L,int i,ElemType *e)
 68  { 
 69      /* 当第i个元素存在时,其值赋给e并返回OK,否则返回ERROR */
 70    int j=1; /* 初始化,j为计数器 */
 71    LinkList p=L->next->next; /* p指向第一个结点 */
 72    if(i<=0||i>ListLength_CL(L)) /* 第i个元素不存在 */
 73      return ERROR;
 74    while(j<i)
 75    { 
 76        /* 顺指针向后查找,直到p指向第i个元素 */
 77      p=p->next;
 78      j++;
 79    }
 80    *e=p->data; /* 取第i个元素 */
 81    return OK;
 82  }
 83 
 84  int LocateElem_CL(LinkList L,ElemType e,Status(*compare)(ElemType,ElemType))
 85  {
 86      /* 初始条件:线性表L已存在,compare()是数据元素判定函数 */
 87    /* 操作结果:返回L中第1个与e满足关系compare()的数据元素的位序。 */
 88    /*           若这样的数据元素不存在,则返回值为0 */
 89    int i=0;
 90    LinkList p=L->next->next; /* p指向第一个结点 */
 91    while(p!=L->next)
 92    {
 93      i++;
 94      if(compare(p->data,e)) /* 满足关系 */
 95        return i;
 96      p=p->next;
 97    }
 98    return 0;
 99  }
100 
101  Status PriorElem_CL(LinkList L,ElemType cur_e,ElemType *pre_e)
102  { 
103      /* 初始条件:线性表L已存在 */
104    /* 操作结果:若cur_e是L的数据元素,且不是第一个,则用pre_e返回它的前驱, */
105    /*           否则操作失败,pre_e无定义 */
106    LinkList q,p=L->next->next; /* p指向第一个结点 */
107    q=p->next;
108    while(q!=L->next) /* p没到表尾 */
109    {
110      if(q->data==cur_e)
111      {
112        *pre_e=p->data;
113        return TRUE;
114      }
115      p=q;
116      q=q->next;
117    }
118    return FALSE;
119  }
120 
121  Status NextElem_CL(LinkList L,ElemType cur_e,ElemType *next_e)
122  { 
123      /* 初始条件:线性表L已存在 */
124    /* 操作结果:若cur_e是L的数据元素,且不是最后一个,则用next_e返回它的后继, */
125    /*           否则操作失败,next_e无定义 */
126    LinkList p=L->next->next; /* p指向第一个结点 */
127    while(p!=L) /* p没到表尾 */
128    {
129      if(p->data==cur_e)
130      {
131        *next_e=p->next->data;
132        return TRUE;
133      }
134      p=p->next;
135    }
136    return FALSE;
137  }
138 
139  Status ListInsert_CL(LinkList *L,int i,ElemType e) /* 改变L */
140  { 
141      /* 在L的第i个位置之前插入元素e */
142    LinkList p=(*L)->next,s; /* p指向头结点 */
143    int j=0;
144    if(i<=0||i>ListLength_CL(*L)+1) /* 无法在第i个元素之前插入 */
145      return ERROR;
146    while(j<i-1) /* 寻找第i-1个结点 */
147    {
148      p=p->next;
149      j++;
150    }
151    s=(LinkList)malloc(sizeof(struct LNode)); /* 生成新结点 */
152    s->data=e; /* 插入L中 */
153    s->next=p->next;
154    p->next=s;
155    if(p==*L) /* 改变尾结点 */
156      *L=s;
157    return OK;
158  }
159 
160  Status ListDelete_CL(LinkList *L,int i,ElemType *e) /* 改变L */
161  {
162      /* 删除L的第i个元素,并由e返回其值 */
163    LinkList p=(*L)->next,q; /* p指向头结点 */
164    int j=0;
165    if(i<=0||i>ListLength_CL(*L)) /* 第i个元素不存在 */
166      return ERROR;
167    while(j<i-1) /* 寻找第i-1个结点 */
168    {
169      p=p->next;
170      j++;
171    }
172    q=p->next; /* q指向待删除结点 */
173    p->next=q->next;
174    *e=q->data;
175    if(*L==q) /* 删除的是表尾元素 */
176      *L=p;
177    free(q); /* 释放待删除结点 */
178    return OK;
179  }
180 
181  Status ListTraverse_CL(LinkList L,void(*vi)(ElemType))
182  {
183      /* 初始条件:L已存在。操作结果:依次对L的每个数据元素调用函数vi()。一旦vi()失败,则操作失败 */
184    LinkList p=L->next->next;
185    while(p!=L->next)
186    {
187      vi(p->data);
188      p=p->next;
189    }
190    printf("\\n");
191    return OK;
192  }

 

以上是关于02循环单链表的主要内容,如果未能解决你的问题,请参考以下文章

检测单链表中循环的开始?

如何用c语言实现单链表的逆置?

删除单链表中的循环[重复]

第02次作业-线性表

数据结构-循环单链表

数据结构链表,看这两篇就足够了(上集,动图版)