数据结构历年考研真题对应知识点(单链表、双链表、循环链表)

目录

2.3线性表的链式表示

2.3.1单链表的定义

【单链表的应用(2009、2012、2013、2015、2016、2019)】

2.3.2单链表上基本操作的实现

【单链表插入操作后地址或指针的变化(2016)】

2.3.3双链表

【双链表中插入操作的实现(2023)】

【循环双链表中删除操作的实现(2016)】

2.3.4循环链表

【循环单链表中删除首元素的操作(2021)】


2.3线性表的链式表示

2.3.1单链表的定义

【单链表的应用(2009、2012、2013、2015、2016、2019)】

线性表的链式存储又称单链表,它是指通过一组任意的存储单元来存储线性表中的数据元素。

为了建立数据元素之间的线性关系,对每个链表结点,除存放元素自身的信息之外,还需要存放一个指向其后继的指针。

单链表中结点类型的描述如下:

cpp 复制代码
typedef struct INode{   //定义单链表结点类型
    ElemType data;          //数据域
    struct LNode *next;  //指针域
}LNode, *LinkList;

利用单链表可以解决顺序表需要大量连续存储单元的缺点,但附加的指针域,也存在浪费存储空间的缺点。

由于单链表的元素离散地分布在存储空间中,因此是非随机存取的存储结构,即不能直接找到表中某个特定结点。查找特定结点时,需要从表头开始遍历,依次查找。

2.3.2单链表上基本操作的实现

【单链表插入操作后地址或指针的变化(2016)】

插入结点操作将值为x的新结点插入到单链表的第i个位置。

先检查插入位置的合法性,然后找到待插入位置的前驱,即第i-1个结点,再在其后插入。其操作过程如图 2.5 所示。

首先査找第 i-1 个结点,假设第 i-1 个结点为*p,然后令新结点*s的指针域指向*p的后继,再令结点*p的指针域指向新插入的结点*s。

cpp 复制代码
bool ListInsert(linklist &L,int i,ElemType e)
    LNode *p=L;              //指针p指向当前扫描到的结点
    int j=0;                 //记录当前结点的位序,头结点是第0个结点
    while(p!=NULL&&j<i-1){   //循环找到第i-1个结点
        p=p->next;
        j++;
}
    if(p==NULL)              //i值不合法
        return false;
    LNode *s=(LNode*)malloc(sizeof(LNode));
    s->data=e;
    s->next=p->next;          //① 
    p->next=s;                //②
    return true;
}

插入时,①和②的顺序不能颠倒,否则,先执行p->next=s 后,指向其原后继的指针就不存在了,再执行 s->next=p->next 时,相当于执行了 s->next=s,显然有误。

本算法主要的时间开销在于查找第 i-1 个元素,时间复杂度为 O(n)。

若在指定结点后插入新结点,则时间复杂度仅为 O(1)。

需注意的是,当链表不带头结点时,需要判断插入位置i是否为1,若是,则要做特殊处理,将头指针工指向新的首结点。

当链表带头结点时,插入位置i为1时不用做特殊处理。

扩展:对某一结点进行前插操作。

前插操作是指在某结点的前面插入一个新结点,后插操作的定义刚好与之相反。

在单链表插入算法中,通常都采用后插操作。以上面的算法为例,先找到第 i-1 个结点,即插入结点的前驱,再对其执行后插操作。由此可知,对结点的前插操作均可转化为后插操作,前提是从单链表的头结点开始顺序查找到其前驱结点,时间复杂度为O(n)。

此外,可采用另一种方式将其转化为后插操作来实现,设待插入结点为*s,将*s插入到*p的前面。我们仍然将*s插入到*p的后面,然后将p->data与 s->data 交换,这样做既满足逻辑关系,又能使得时间复杂度为O(1)。该方法的主要代码片段如下:

cpp 复制代码
s->next=p->next; //修改指针域,不能颠倒
p->next=s;
temp=p->data;    //交换数据域部分
p->data=s->data;
s->data=temp;

2.3.3双链表

【双链表中插入操作的实现(2023)】

在双链表中p所指的结点之后插入结点*s,其指针的变化过程如图 2.10 所示。

插入操作的代码片段如下:

cpp 复制代码
s->next=p->next;     //将结点*s插入到结点*p之后
p->next->prior=s;
s->prior=p;
p->next=s;

上述代码的语句顺序不是唯一的,但也不是任意的,①步必须在④步之前,否则*p的后继结点的指针就会丢掉,导致插入失败。

【循环双链表中删除操作的实现(2016)】

删除双链表中结点*p的后继结点*q,其指针的变化过程如图 2.11所示。

删除操作的代码片段如下:

删除双链表中结点*p的后继结点*q;

删除操作的代码片段如下:

cpp 复制代码
p->next=q->next;
q->next->prior=p;
free(q);            //释放结点空间

在建立双链表的操作中,也可采用如同单链表的头插法和尾插法,但在操作上需要注意指针的变化和单链表有所不同。

2.3.4循环链表

【循环单链表中删除首元素的操作(2021)】

循环单链表的插入、删除算法与单链表的几乎一样,所不同的是若操作是在表尾进行,则执行的操作不同,以让单链表继续保持循环的性质。当然,正是因为循环单链表是一个"环",所以在任何位置上的插入和删除操作都是等价的,而无须判断是否是表尾。

**在单链表中只能从表头结点开始往后顺序遍历整个链表,而循环单链表可以从表中的任意一个结点开始遍历整个链表。**有时对循环单链表不设头指针而仅设尾指针,以使得操作效率更高其原因是,若设的是头指针,对在表尾插入元素需要O(n)的时间复杂度,而若设的是尾指针r,r->next即为头指针,对在表头或表尾插入元素都只需要 O(1)的时间复杂度。

相关推荐
木向38 分钟前
leetcode92:反转链表||
数据结构·c++·算法·leetcode·链表
阿阿越40 分钟前
算法每日练 -- 双指针篇(持续更新中)
数据结构·c++·算法
小爬虫程序猿3 小时前
如何利用Python解析API返回的数据结构?
数据结构·数据库·python
pianmian17 小时前
python数据结构基础(7)
数据结构·算法
ChoSeitaku10 小时前
链表交集相关算法题|AB链表公共元素生成链表C|AB链表交集存放于A|连续子序列|相交链表求交点位置(C)
数据结构·考研·链表
偷心编程10 小时前
双向链表专题
数据结构
香菜大丸10 小时前
链表的归并排序
数据结构·算法·链表
jrrz082810 小时前
LeetCode 热题100(七)【链表】(1)
数据结构·c++·算法·leetcode·链表
@小博的博客11 小时前
C++初阶学习第十弹——深入讲解vector的迭代器失效
数据结构·c++·学习
泉崎13 小时前
11.7比赛总结
数据结构·算法