文字描述

  之前的链表(单链表、循环链表)的链式存储结构中只有一个指示直接后继的指针域。由此,从某个结点出发只能顺指针往后寻查其他结点。若要寻查结点的直接前驱,则需从表头指针出发。即单链表中,NextElem的执行时间为1,而PriorElem的执行时间为n。为了克服单链表这种缺点,可利用双向链表。

  在双向链表中,每个节点有两个指针域,一个指向其直接后继,一个指向其直接前驱。

示意图

算法分析

插入、删除、查找、求后继等同单链表。但是求前驱不一样,其时间复杂度为1。

代码实现

 //
// Created by lady on 19-1-31.
// #include <stdlib.h>
#include <stdio.h>
#include <string.h> //线性表的双向循环链表存储结构
typedef struct ElemType{
char data[];
}ElemType;
typedef struct DuLNode{
ElemType e;
struct DuLNode *prior;
struct DuLNode *next;
}DuLNode, *DuLinkList; //构造一个空的双向循环链表
static int InitList_DuL(DuLinkList *L)
{
(*L) = (DuLinkList)malloc(sizeof(DuLNode));
(*L)->prior = (*L);
(*L)->next = (*L);
memset((*L)->e.data, , sizeof((*L)->e.data));
return ;
} //销毁L
static int Destory_DuL(DuLinkList *L)
{
DuLNode *p = NULL;
DuLNode *q = NULL;
p = (*L)->next;
while(p){
q = p;
p = p->next;
free(q);
if(p == (*L))
break;
}
free(*L);
(*L) = NULL;
return ;
} //返回双向链表L中的第i个位置的元素
static DuLNode *GetElemP_DuL(DuLinkList *L, int i)
{
if(L == NULL){
return NULL;
}
DuLNode *p = (*L)->next;
int index = ;
while(index<i && p){
index += ;
p = p->next;
}
if(index == i && p){
return p;
}else{
return NULL;
}
} //在L的指定位置i的元素前面插入元素e
static int ListInsert_DuL(DuLinkList *L, int i, ElemType e)
{
DuLNode *p = NULL;
DuLNode *new = NULL;
if((p=GetElemP_DuL(L, i)) == NULL){
return -;
}
if((new = (DuLNode*)malloc(sizeof(DuLNode))) == NULL){
return -;
}
new->e = e;
new->prior = p->prior;
p->prior->next = new;
new->next = p;
p->prior = new;
return ;
} //删除L中的第loc个数据元素,并将被删元素的值存放在e中
static int ListDelete_DuL(DuLinkList *L, int i, ElemType *e)
{
DuLNode *p = NULL;
if((p=GetElemP_DuL(L, i)) == NULL){
return -;
}
(*e) = p->e;
p->prior->next = p->next;
p->next->prior = p->prior;
free(p);
return ;
} //依次对L的每个数据元素调用函数fun
static int listTraverse_DuL(DuLinkList *L, int (*fun)(ElemType,int), char note[])
{
printf("遍历双向循环链表%s:", note);
DuLNode *p = NULL;
p = (*L)->next;
int i = ;
while(p){
fun(p->e, i);
i+=;
p = p->next;
if(p == (*L))
break;
}
printf("\n");
return ;
} static int print(ElemType e, int loc)
{
printf("%3d=%-10s", loc, e.data);
} //创建一个长度为n的双向链表
static int CreateList_DuL(DuLinkList *L, int n, char note[])
{
printf("创建一个长度为%d的双向循环链表%s!\n", n, note);
InitList_DuL(L);
ElemType e;
int i = ;
for(i=; i<n; i++){
printf("输入第%d个元素:", i+);
scanf("%s[^\\n]", e.data);
ListInsert_DuL(L, i+, e);
}
return ;
} int main(int argc, char *argv[])
{
ElemType e;
DuLinkList L;
int location=; CreateList_DuL(&L, , "L"); listTraverse_DuL(&L, print, "L"); printf("输入插入元素的位置:");
scanf("%d", &location);
printf("输入插入元素的数据:");
scanf("%s[^\\n]", e.data);
ListInsert_DuL(&L, location, e);
listTraverse_DuL(&L, print, "L"); printf("输入删除元素的位置:");
scanf("%d", &location);
ListDelete_DuL(&L, location, &e); printf("位于%d的元素%s已经从双向循环链表中被删除!\n", location, e.data);
listTraverse_DuL(&L, print, "L"); Destory_DuL(&L);
printf("双向链表已经被销毁!\n");
return ;
}

双向链表

代码运行

/home/lady/CLionProjects/untitled/cmake-build-debug/untitled
创建一个长度为5的双向循环链表L!
输入第1个元素:c
输入第2个元素:b
输入第3个元素:e
输入第4个元素:g
输入第5个元素:f
遍历双向循环链表L: 1=c 2=b 3=e 4=g 5=f
输入插入元素的位置:3
输入插入元素的数据:test
遍历双向循环链表L: 1=c 2=b 3=test 4=e 5=g 6=f
输入删除元素的位置:4
位于4的元素e已经从双向循环链表中被删除!
遍历双向循环链表L: 1=c 2=b 3=test 4=g 5=f
双向链表已经被销毁! Process finished with exit code 0

线性表->链式存储->双向链表的更多相关文章

  1. 线性表->链式存储->循环链表

    文字描述 循环链表是另一种形式的链式存储结构.它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环.由此,从表中任一结点出发均可找到表中其他结点. 示意图 算法分析 插入.删除.查找等同单 ...

  2. 线性表->链式存储->线形链表(单链表)

    文字描述: 为了表示前后两个数据元素的逻辑关系,对于每个数据元素,除了存储其本身的信息之外(数据域),还需存储一个指示其直接后继的信息(即直接后继的存储位置,指针域). 示意图: 算法分析: 在单链表 ...

  3. C语言实现线性表(链式存储方式)

    #include <stdio.h> #include <stdlib.h> //提供malloc()原型 typedef struct LNode *List; typede ...

  4. C语言 线性表 链式表结构 实现

    一个单链式实现的线性表 mList (GCC编译). /** * @brief 线性表的链式实现 (单链表) * @author wid * @date 2013-10-21 * * @note 若代 ...

  5. javascript实现数据结构:线性表--线性链表(链式存储结构)

    上一节中, 线性表的顺序存储结构的特点是逻辑关系上相邻的两个元素在物理位置上也相邻,因此可以随机存取表中任一元素,它的存储位置可用一个简单,直观的公式来表示.然后,另一方面来看,这个特点也造成这种存储 ...

  6. 算法与数据结构(一) 线性表的顺序存储与链式存储(Swift版)

    温故而知新,在接下来的几篇博客中,将会系统的对数据结构的相关内容进行回顾并总结.数据结构乃编程的基础呢,还是要不时拿出来翻一翻回顾一下.当然数据结构相关博客中我们以Swift语言来实现.因为Swift ...

  7. c数据结构 -- 线性表之 复杂的链式存储结构

    复杂的链式存储结构 循环链表 定义:是一种头尾相接的链表(即表中最后一个结点的指针域指向头结点,整个链表形成一个环) 优点:从表中任一节点出发均可找到表中其他结点 注意:涉及遍历操作时,终止条件是判断 ...

  8. [置顶] ※数据结构※→☆线性表结构(queue)☆============优先队列 链式存储结构(queue priority list)(十二)

    优先队列(priority queue) 普通的队列是一种先进先出的数据结构,元素在队列尾追加,而从队列头删除.在优先队列中,元素被赋予优先级.当访问元素时,具有最高优先级的元素最先删除.优先队列具有 ...

  9. C++编程练习(2)----“实现简单的线性表的链式存储结构“

    单链表采用链式存储结构,用一组任意的存储单元存放线性表的元素. 对于查找操作,单链表的时间复杂度为O(n). 对于插入和删除操作,单链表在确定位置后,插入和删除时间仅为O(1). 单链表不需要分配存储 ...

随机推荐

  1. # 20175333曹雅坤《Java程序设计》第1周学习总结

    教材学习内容总结 1.学习第一章PPT,安装JRE,JDK并配置path环境参数 2.在windows上使用dos命令运行教材第一章代码Hello.java和People.java 3.下载使用git ...

  2. AGC电路以及AD8347正交解调芯片

    1.AGC电路的工作原理 1.1AGC电路的用途 随着电磁环境的日益恶化, 不同频段电磁信号之间的相互串扰, 以及可能出现的人为干扰, 将会导致接收机输入端口的信号动态范围较大, 一旦出现电路饱和或是 ...

  3. vue中怎么实现获取当前点击对象this

    应用场景 在评论列表中,有很多条评论(通过循环出来的评论列表),评论的文字有多跟少,默认展示2行超出显示点击查看更多,,要点击查看更多对当前的这条评论进行全部评论展示! 问题描述 要是在传统的点击事件 ...

  4. PHP生成HTML文件, SummerHtml

    2018-6-27 20:13:04 星期三 作用: 用PHP生成HTML文档, 支持标签嵌套缩进 起因: 这个东西确实也是心血来潮写的, 我很满意里边的实现缩进的机制, 大家有用到的可以看看 现在都 ...

  5. 51nod--1134 最长递增子序列 (动态规划)

    题目: 给出长度为N的数组,找出这个数组的最长递增子序列.(递增子序列是指,子序列的元素是递增的) 例如:5 1 6 8 2 4 5 10,最长递增子序列是1 2 4 5 10. Input 第1行: ...

  6. storage和memory

    memory:使用的是值传递,默认使用的是memory,传递的是值 storage:引用传递,传过来的是指针,后面一定要加上internal,private pragma solidity ^; co ...

  7. cdh zookeeper 不断拒绝连接

    测试hiveserver2高可用的时候,需要登录zookeeper检查hiveserver2是否在zookeeper中注册 执行 zookeeper-client 不断的拒绝连接 -- ::, [my ...

  8. RDay1-Problem 1 A

    题目描述 给定一个长度为n的正整数序列a[i],计算出有多少个i<j的数对,a[i]+a[j]为二的次幂,也就是说存在一个正整数x满足a[i]+a[j]==2^x. 输入 输入文件A.in. 第 ...

  9. 微信小程序换皮肤,动态切换菜单栏和导航栏的样式,动态修改TabBar和NavigationBar

    在做微信小程序换皮肤的时候,需要动态修改菜单栏(TabBar)和导航栏(NavigationBar) 但是在小程序中它们的样式是写在app.json里面,而且app.json是静态编译,运行时哪怕你修 ...

  10. Vue 增删改查 demo

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...