(四)单链表
最后更新于:2022-04-01 09:40:10
> 上一篇博客学习了顺序表,最后也说明了顺序表属于静态存储,数据元素的个数不能自由的扩充。为了解决这个问题我们引入了链表
### 链表存储结构
结点在存储器中的位置是任意的,即逻辑上相邻的数据元素在物理上不一定相邻,因此线性表的链式表示又称为非顺序映像或链式映像。
### 各个结点有两个域组成:
![结点构成](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1e7aa98.jpg "")
* 数据域:存储元素数值数据
* 指针域:存储直接后继结点的存储位置
### 名词解析
![结点](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1e89075.jpg "")
1. 结点:数据元素的存储映像。有数据域和指针域两部分组成。
2. 链表:n个结点由指针链组成一个链表。它是线性表的链式存储映像,称为线性表的链式存储结构
3. 单链表、双链表、循环链表:
* 结点只有一个指针域的链表,称为单链表或线性链表
* 有两个指针域的链表,称为双链表
* 首尾相接的链表称为循环链表
4. 头指针是指向链表中第一个结点的指针
5. 首元结点是指链表中存储第一个数据元素a1的结点
6. 头结点是在链表的首元结点之前附设的一个结点;数据域内只放空表标志和表长等信息(可无)
当头结点的指针域为空时表示空表
头结点不能计入链表长度值
### 单链表的定义和实现
非空表
空表
![表](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1e9882f.jpg "")
单链表是由表头唯一确定,因此单链表可以用头指针的名字来命名
若头指针名是L,则把链表称为表L
单链表的存储结构定义
~~~
typedef struct LNode
{
ElemType data; //数据域
struct LNode *next; //指针域
}LNode,*LinkList; //*LinkList为Lnode类型的指针
~~~
注意:
指针变量p:表示结点地址
结点变量*p:表示一个结点
### 单链表基本操作的实现
#### 1.初始化(构造一个空表)
【算法思想】
(1)生成新结点作为头结点,用头结点L指向头结点。
(2)头结点的指针域置空。
【算法描述】
~~~
Status InitList_L(LinkList &L)
{
L=new LNode;
L->next=NULL;
return OK;
}
~~~
#### 2.查找
##### (1)按序号查找
【算法思想】
从链表的第一个结点(L->next)开始顺着链域扫描,用指针p指向当前扫描到的结点,p的初始值指向第1个结点(p=L->next)。用j做计数器,累计当前扫描过的结点数,j的初值为1,当p指向扫描到的下一个结点时,计数器j相应加1.当j=i时,p所指向的结点是想要找的第i个结点。
【算法描述】
~~~
Status GetElem_L(LinkList L,int i,ElemType &e)
{
p=L->next;j=1;
while(p&&j<i)
{
p=p->next;
++j;
}
if(!p||j>i)
return ERROR;
e=p->data;
return OK;
}
~~~
#### 按值查找
【算法思想】
链表中查找其值与给定e相等的数据元素的过程和顺序表类似,从第一个结点起,依次和e相比较,如果找到一个其值与e相等的数据元素,则返回其在链表中的“位置”;如果查遍整个链表都没有找到其值和e相等的元素,则返回“NULL”。因此需要设置一个指针变量p顺链扫描,直至p为“NULL”,或者p->data和e相等为止。
【算法描述】
~~~
LNode *LocateElem_L(LinkList L,ElemType e)
{
p=L->next;
while(p&&p->data!=e)
{
p=p->next;
}
return p;
}
~~~
### 3.插入
【算法思想】
![插入](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1eadac4.jpg "")
将值为e的新结点插入到表的第i个结点的位置上,即插入到结点ai-1与ai之间,分为一下几步:
1. 找到结点ai-1并由指针p指向该结点。
1. 生成一个新结点*s。
1. 将新结点*s的数据域置为e。
1. 将新结点*s的指针域指向结点ai。
1. 令结点ai-1的指针域指向新结点*s。
【算法描述】
~~~
Status ListInsert_L(LinkList &L,int i,ElemType e)
{
p=L;j=0;
while(p&&j<i-1)
{
p=p->next;
++j;
}
if(!p||j>i-1)
return ERROR;
s=new LNode;
s->data=e;
s->next=p->next;
p->next=s;
return OK;
}
~~~
### 4.删除
【算法思想】
要删除单链表的第i个结点ai,分以下几步:
![删除](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1ec04a5.jpg "")
1. 找到结点ai-1并由指针p指向该结点。
2. 临时保存待删除结点ai的地址在q中,以备释放。
3. 令p->next指向ai的直接后继结点。
4. 将待删除结点ai的值保留在e中
5. 释放结点ai的空间。
【算法描述】
~~~
Status ListDelete_L(LinkList &L,int i,ElemType &e)
{
p=L;j=0;
while(p->next&&j<i-1)
{
p=p->next;++j;
}
if(!(p->next)||j>i-1)
return ERROR;
q=p->next;
e=q->data;
delete q;
return OK;
}
~~~
### 5.创建单链表
#### (1)前插法
【算法思想】
前插法是通过将新结点逐个插入链表的头部(头结点之后)来创建链表。首先建立只有一个头结点的空链表,每读入一个数据元素则申请一个新结点,并将新结点插入到头结点之后。
![前插法](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1edbfcd.jpg "")
【算法描述】
~~~
void CreateLst_F(LinkList &L,int n)
{
L=new LNode;
L->next=NULL;
for(i=n;i>0;--i)
{
p=new LNode;
cin>>p->data;
p->next=L->next;L->next=p;
}
}
~~~
#### (2)后插法
【算法思想】
后插法是通过将新结点逐个插入到链表的尾部来创建链表。同前插法一样,首先要创建一个只有头结点的空链表L,不同的是,为了使新结点能够插入到表尾,需要增加一个尾指针r指向链表的尾结点,初始化时,r同L均指向头结点。每读入一个数据元素则申请一个新结点,将新结点插入到尾结点*r之后,然后使r指向新的尾结点。
![后插法](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1f042b2.jpg "")
【算法描述】
~~~
void CreateList_L(LinkList &L,int n)
{
L=new LNode;
L->next=NULL;
r=L;
for(i=0;i<n;++i)
{
p=new LNode;
cin>>p->data;
p->next=NULL;r->next=p;
r=p;
}
}
~~~
### 循环链表
循环链表是表中最后一个结点的指针域指向头结点,整个链表形成一个环。
![l两个链表合并](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-25_56ce7d1f210e1.jpg "")
两个线性表合并成一个表,将第一个表的尾指针指向第二个表的第一个结点,第二个表的尾指针指向第一个表的头结点,然后释放第二个表的头结点。
~~~
p=B->next->next;
B->next=A->next;
A->next=p;
~~~