直接贴出完整代码,每个函数的功能及部分代码的解释都在注释中,代码亲测可行

/*
2018.8.15
注意三点:
1.不要将循环写成if    //很尴尬,主要是我犯了这个错误,找了半天还没找出来,第二天看的时候才发现,非常的尴尬
2.循环链表的判空操作是 p->rear != *L
3.p = *L,循环体中用p->rear做条件 这种写法便于对当前结点的前一结点操作,插入、删除、修改操作使用这种形式
p = *L->rear,循环体中用p做条件 这种写法便于对当前结点操作,查找、遍历使用这种形式
       4.双向链表在插入与删除时,要处理好前驱指针和后继指针,切不可遗忘。尤其是插入时,要将新点之后的结点的前驱指向新结点,这里容易被忽略
*/ #include <stdio.h>
#include<malloc.h>
#include<stdlib.h>
#include<time.h>
typedef int ElemType; //元素类型
typedef int Status; //返回类型
#define OK 1; //函数运行成功返回值
#define ERROR 0; //函数运行失败返回值 typedef struct Node
{
struct Node *pront; //指向前一结点的指针
ElemType date; //结点元素
struct Node *rear; //指向后一结点的指针
}Node; typedef struct Node *LoopList; //创建链表 /*
循环双向链表的插入操作
*/
Status LoopInsert(LoopList *L, int i, ElemType e)
{
int j = ; //计数器,记录当前位置
LoopList r = *L; //指向L //此种写法便于对当前结点的上一结点进行操作
LoopList s; //用于创建新结点
// while(r->rear && j < i) //判非空 判位置索引合理 //循环链表的判断非空不是这样的 2018.8.15
while(r->rear != *L && j < i)
{
r = r->rear; //指针后移
j++; //计数器加1
}
if(r->rear == *L || j > i) //判空 判位置索引不合理
return ERROR;
s = (LoopList)malloc(sizeof(Node)); //开辟新结点s
s->date = e; //为s的数据域赋值
r->rear->pront = s; //使插入结点后的下一结点的前驱指向s
s->rear = r->rear; //使s的后继结点等于r的后继结点
s->pront = r; //使s的前驱结点等于r
r->rear = s; //使r得后继结点等于s
return OK;
} /*
循环双向链表的删除操作
*/
Status DelList(LoopList *L, int i, ElemType *e)
{
int j = ; //记录当前位置
LoopList r = *L; //指向链表
LoopList s; //用于释放要删除的结点
while(r->rear != *L && j < i) //判非空 判索引位置有效
{
r = r->rear; //指针后移
j++; //计数器加1
}
if(r->rear == *L || j > i) //判空 判索引位置无效
return ERROR;
s = r->rear; //使s指向r
*e = s->date; //将要删除的结点数据赋值给e
r->rear = s->rear; //使r的后继结点等于r的后继结点
s->rear->pront = r; //使s的后继的前驱结点等于r
free(s); //释放s结点
return OK;
} /*
循环双链表的修改操作
*/
Status UpdateList(LoopList *L, int i, ElemType e)
{
int j = ; //记录当前位置
LoopList r = (*L)->rear; //指向第一个结点 //此种写法便于岁当前结点操作
while(r != *L && j < i) //判非空 判位置索引有效
{
r = r->rear; //指针后移
j++; //计数器加1
}
if(r == *L || j > i) //判空 判位置索引无效
return ERROR;
r->date = e; //使r的数据域等于 e
return OK;
} /*
循环双链表的查找
*/
Status GetElem(LoopList L, int i, ElemType *e)
{
int j = ; //计数器
LoopList r = L->rear; //指向第一个结点
while(r != L && j < i) //判非空 判位置索引有效
{
r = r->rear; //指针后移
j++; //计数器加1
}
if(r == L || j > i) //判空 判位置索引无效
return ERROR;
*e = r->date; //将r的数据域内容赋值给e
return OK;
} /*
循环双链表的正序遍历
*/
void PrintList1(LoopList L)
{
int j = ;
LoopList r = L->rear; //指向L第一个结点
if(r == L) //判空
printf("表空\n");
while(r != L) //判非空
{
printf("第%d个结点的数据是%d\n",j,r->date);
r = r->rear;
j++;
}
printf("\n");
} /*
循环双链表的倒序遍历
*/
void PrintList2(LoopList L)
{
int j = ;
LoopList r = L->pront; //指向L倒数第一个结点
if(r == L) //判空
printf("表空\n");
while(r != L) //判非空
{
printf("倒数第%d个结点的数据是%d\n",j,r->date);
r = r->pront;
j++;
}
printf("\n");
} /*
循环双链表的创建
*/
Status CreatList(LoopList *L, int n)
{
int i; //计数器
(*L) = (LoopList)malloc(sizeof(Node)); //创建头结点
LoopList s,q; //s用于开辟新结点,q指向表尾
srand(time()); //初始化随机数种子
q = *L; //指向表尾结点
for(i = ; i < n; i++)
{
s = (LoopList)malloc(sizeof(Node)); //开辟新结点
s->date = rand() % + ; //为新结点赋值
q->rear = s; //让表尾结点的后继结点指向新结点
s->pront = q; //让新结点的前驱结点指向q
q = s; //表尾结点指针后移
}
q->rear = *L; //使表尾结点的后继指针指向头结点
(*L)->pront = q; //头指针的前驱结点指向表尾结点
return OK;
} void main()
{
LoopList L = NULL; //创建链表L
int i, e; //i为元素位置,e为元素内容 while(true)
{
printf("请选择对线性链表的操作:\n");
printf("1.创建\n");
printf("2.插入\n");
printf("3.删除\n");
printf("4.查找\n");
printf("5.修改\n");
printf("6.正序输出\n");
printf("7.倒序输出\n");
printf("8.退出\n");
int a;
scanf("%d", &a);
switch(a)
{
case :
printf("请输入需要创建元素的个数:");
scanf("%d", &i);
if(CreatList(&L, i))
printf("创建成功\n");
else
printf("创建失败\n");
break;
case :
printf("请输入需要插入的位置:");
scanf("%d", &i);
printf("请输入需要插入的元素:");
scanf("%d", &e);
if(LoopInsert(&L, i, e))
printf("插入成功\n");
else
printf("插入失败\n");
break;
case :
printf("请输入需要删除的位置:");
scanf("%d", &i);
if(DelList(&L, i, &e))
printf("删除成功\n");
else
printf("删除失败\n");
break;
case :
printf("请输入需要查找的位置:");
scanf("%d", &i);
GetElem(L, i, &e);
printf("第%d个元素为%d\n",i,e);
break;
case :
printf("请输入需要修改的位置:");
scanf("%d", &i);
printf("请输入新的的元素:");
scanf("%d", &e);
if(UpdateList(&L, i, e))
printf("修改成功\n");
else
printf("修改失败\n");
break;
case :
if(L == NULL)
{
printf("表还未创建\n");
break;
}
PrintList1(L);
break;
case :
if(L == NULL)
{
printf("表还未创建\n");
break;
}
PrintList2(L);
break;
case :
return;
default:
printf("选择错误\n");
break;
}
}
}

循环双向链表-C语言实现的更多相关文章

  1. 纯C语言实现循环双向链表创建,插入和删除

    #include <stdio.h> #include <stdlib.h> typedef int ElemType; typedef struct DLNode{ Elem ...

  2. (续)线性表之双向链表(C语言实现)

    在前文实现单向链表的基本操作下,本文实现双向链表的基本操作. 双向链表与单链表差异,是双向链表结点中有前向指针和后向指针.所以在插入和删除新结点元素时候不见要考虑后向指针还要考虑前向指针. 以下是双向 ...

  3. 字符串循环右移-c语言

    一个长度为len的字符串,对其循环右移n位 [期望]char str[] = "abcdefg";右移3次后,变成"efgabcd" [思路] 思路1. 如果用 ...

  4. 数据结构 双向链表 C语言实现

    dlist.h #ifndef __dList_H #define __dlist_H typedef int Item; typedef struct Node *PNode; typedef PN ...

  5. 定长循环队列C语言实现

    #ifndef _CONST_H_#define _CONST_H_ #include <stdio.h>#include <stdlib.h> typedef enum { ...

  6. 【数据结构】循环队列 C语言实现

    "Queue.h" #include "Queue.h" #include <stdio.h> #include <stdlib.h> ...

  7. 在django中使用循环与条件语言

    {% if not Article_type_id %} <li class="active"><a href="/">全部</a ...

  8. for循环(C语言型)举例

  9. for循环(C语言型)流程

随机推荐

  1. Python文件处理:创建、打开、追加、读、写

    在Python中,不需要导入外部库来读取和写入文件.Python为创建.写入和读取文件提供了内置的函数. 在本文中,我们将学习 如何创建文本文件 如何将数据附加到文件中 如何读取文件 如何逐行读取文件 ...

  2. %%%GXZ大佬回关

  3. 原生JS实现栈结构

    1. 前言 栈,是一种遵从后进先出(LIFO,Later-In-First-Out)原则的有序集合.新添加的元素都保存在栈的一端,称作栈顶,另一端叫做栈底.在栈中,新元素都靠近栈顶,旧元素都靠近栈底. ...

  4. svg路径蒙版动画

    svg路径蒙版动画,是比较实用的一种动画效果,能够绘制如下图所示的动画. 接下来细说这样的动画是如何做成的: 1.准备工作 2.SVG路径动画 3.SVG路径蒙版动画 4.复杂图形的编辑技巧 1.准备 ...

  5. 大数据之路week01--day02我实在时被继承super这些东西搞的头疼,今天来好好整理以下。

    这一周的第一天的内容是面向对象的封装,以及对方法的调用.实在时没法单独拿出来单说,就结合这一节一起说了. 我实在是被继承中的super用法给弄的有点晕,程序总是不能按照我想的那样,不是说结果,而是实现 ...

  6. 「CF630C」Lucky Numbers

    更好的阅读体验 Portal Portal1: Codeforces Portal2: Luogu Description The numbers of all offices in the new ...

  7. Unicode字符串和非Unicode字符串

    什么是Unicode? Unicode(统一码.万国码.单一码)是计算机科学领域里的一项业界标准,包括字符集.编码方案等.Unicode 是为了解决传统的字符编码方案的局限而产生的,它为每种语言中的每 ...

  8. HTTPS加密流程理解

    HTTPS加密流程 由于HTTP的内容在网络上实际是明文传输,并且也没有身份验证之类的安全措施,所以容易遭到挟持与攻击 HTTPS是通过SSL(安全套接层)和TLS(安全传输协议)的组合使用,加密TC ...

  9. count的一些用法

    count(*)包括了所有的列,相当于行数,在统计结果的时候,不会忽略列值为NULL  count(1)包括了所有列,用1代表代码行,在统计结果的时候,不会忽略列值为NULL  count(列名)只包 ...

  10. LeetCode18. 四数之和

    LeetCode18. 四数之和 给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d ,使得 a + b + c + d 的值 ...