c语言链表专题

参考博客
https://www.cnblogs.com/clover-toeic/p/3793131.html

C语言通用双向循环链表操作函数集
一 概念
链表是一种物理存储单元上非连续、非顺序存储结构,数据元素的逻辑顺序通过链表中的指针链接次序实现。链表由一系列存储结点组成,结点可在运行时动态生成。每个结点均由两部分组成,即存储数据元素的数据域存储相邻结点地址的指针域。当进行插入或删除操作时,链表只需修改相关结点的指针域即可,因此相比线性表顺序结构更加方便省时。

 链表可分为单链表(Singly Linked List)、双向链表(Doubly Linked List)和循环链表(Circular List)等。一个单链表结点仅包含一个指向其直接后继结点的指针,因此当需要操作某个结点的直接前驱结点时,必须从单链表表头开始查找。

 双向链表和循环链表均为单链表的变体。通常创建双向循环链表以综合利用两者的优点。

1.1 双向链表
双向链表的每个结点除含有数据域外,还有两个指针域,分别指向直接前驱结点和直接后继结点。因此,从双向链表中的任一结点开始,均可方便地访问其前驱结点和后继结点。双向链表的结点结构示意图如下所示:

在这里插入图片描述

图1 双向链表的结点结构

 其中,Data为结点存储的数据元素,prev指针指向该结点的前驱结点,next指针指向该结点的后继结点。双向链表通常含有一个表头结点,亦称哨兵结点(Sentinel Node),用于简化插入和删除等操作。带头结点的非空双向链表如下图所示:

在这里插入图片描述

图2 带头结点的非空双向链表

 图中,表头指针dhead指向表头结点Head,该结点的前驱指针为空;结点C为表尾结点,其后继指针为空。除表头结点和表尾结点外,对指向双向链表任一结点的指针p,满足下面的关系:

p = p->prev->next = p->next->prev

即当前结点前驱的后继是自身,其后继的前驱也是自身。

 链表有查找、插入和删除三种基本操作。双向链表也不例外。
  1. 查找操作

    在带表头的双向链表中查找数据域为一特定值的某个结点时,可从表头结点开始向后依次匹配各结点数据域的值,若与特定值相同则返回指向该结点的指针,否则继续往后遍历直至表尾。

    1. 插入操作

    假设指针p和q指向双向链表中的两个前后相邻结点,将某个新结点(指针为s)插到p和q之间,其过程及C语言描述如下图所示:
    在这里插入图片描述
    注意,结点前驱后继指针的操作顺序并非唯一,但必须保证最后才对p->next或q->prev赋值(操作➃),否则会“丢失”p的后继结点或q的前驱结点

可见,若相邻结点指针p、q均已知,则在p和q之间插入新结点s时,只需依次将s的前驱指针指向p,s的后继指针指向q,p的后继指针指向s,q的前驱指针指向s。即:

① s->prev = p;

② s->next = q;

③ p->next = s;

④ q->prev = s;

双向链表中p和q->prev指向同一结点,因此上述步骤等效于图3中q“视角”的第二种插入顺序。为便于记忆,可想象孩子(s)先后去拉爸爸§和妈妈(q)的手,爸爸§妈妈(q)再先后拉住孩子(s)的手。

  1. 删除操作
    删除某个结点,其实就是插入某个结点的逆操作。还是对于双向循环链表,要在连续的三个结点s,p,q中删除p结点,只需把s的右链域指针指向q,q的左链域指针指向s,并收回p结点即可。

    假设指针p、s和q指向双向链表中的三个前后相邻结点,删除结点s的过程及C语言描述如下图所示:
    在这里插入图片描述
    可见,删除时只需将p的后继指针指向q,q的前驱指针指向p,并回收结点s即可。

1.2 循环链表
将单链表尾结点的指针域指向第一个结点或表头结点,即构成单向循环链表,简称循环链表。从循环链表中任一结点单向出发,均可找到链表中其他结点。

 借助表头结点可统一空表和非空表的运算,因此循环链表中往往加入表头结点。带头结点的循环链表如下图所示:
![在这里插入图片描述](https://img-blog.csdnimg.cn/20190424202353803.jpg?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80NDg0MDc0NA==,size_16,color_FFFFFF,t_70)
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值