前言:上期我们介绍了算法的复杂度,知道的算法的重要性同时也了解到了评判一个算法的好与坏就去看他的复杂度(主要看时间复杂度),这一期我们就从顺序表和链表开始讲起。
文章目录
- 一,顺序表
- 1,线性表
- 2,顺序表
- 3,顺序表的分类
- 4,动态顺序表的实现
- 1,增
- 1,头插
- 2,尾插
- 2,在指定位置前插入数据
- 2,删
- 1,头删
- 2,尾删
- 3,删除pos位置(指定位置)的数据
- 3,查
- 4,改
- 5,销毁
- 二,顺序表的总结和问题的思考
- 三,链表
- 1,什么是链表?
- 2,什么是结点?
- 3,链表的分类
- 4,单链表的实现
- 1,增
- 1,头插
- 2,尾插
- 3,在指定位置之前插入数据
- 4,在指定位置之后插入数据
- 1,删
- 1,头删
- 2,尾删
- 3,删除pos位置之前的结点
- 4,删除pos位置之后的结点
- 3,查
- 4 ,链表的销毁
一,顺序表
在介绍顺序表之前我们先引出一个概念:线性表
1,线性表
什么是线性表?
线性表(linear list)是n个具有相同特性的数据元素的有限序列。
线性表是⼀种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串…。
· 线性表在逻辑上是线性结构也就是连续的一条直线;但在物理结构上并不一定是连续的,线性表在物理上储存时通常以数组或者链式结构的形式储存。
2,顺序表
什么是顺序表?
概念:顺序表是⽤⼀段物理地址连续的存储单元依次存储数据元素的线性结构,⼀般情况下采⽤数组存储。
通过概念不难发现顺序表它也是一个线性表,那就有线性表所对应的性质。那么线性表既然采用数组存储那么就说明它的底层就是一个数组,我们对于顺序表的操作就是对数组的操作。
从上图我们可以看地出来,顺序表地逻辑结构是线性的,物理结构也是线性的。
逻辑结构是认为构想出来的的,比如上图就是人为想象出来的。
而物理结构是线性的是因为顺序表底层是数组,数组它本身就是连续存放数据的。
看到这有人就会有疑问了说顺序表和数组到底有什么区别呢?
首先数组是一个可以存放n个相同特性元素的集合,顺序表底层使用的是数组所以顺序也可以存放n个具有相同特性元素的集合,但是顺序表可以对它的底层数组执行一些操作比如对数组内部的数据进行增,删,查,改等操作。
所以可以理解为顺序表是对数组的一个封装,可以直接操作数组。
3,顺序表的分类
顺序表分为动态顺序表和静态顺序表两种。
静态顺序表 即使用定长数组
我们很容易发现它的缺点就是数组空间的问题,如果我们初始时的空间给大了却只存储很少的数据就造成了空间的浪费,如果我们初始时给的空间太小如果要存储大量的数据有会存在空间不足的问题,所以在实现顺序表时一般不采用静态顺序表的方式。
再来看下一种
动态顺序表 即空间大小可以随时变化
将数组改成指针的形式之后就可以通过malloc,realloc和calloc函数来扩展空间了,就很好的解决了静态顺序表对于空间的担忧。所以在实现顺序表的时候就使用动态顺序表。
注意:如果上面代码看不懂的话可以去看看博主写的C语言有关结构体,内存函数,指针的内容,因为数据结构就是使用他们来实现的!!!
4,动态顺序表的实现
首先需要定义一个顺序表在哪定义呢?
数据结构中的结构就是结构体,当然要在结构体内部定义了。
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
//定义一个顺序表
typedef int SLtype;
typedef struct sqelist
{
SLtype* arr;//顺序表底层的数组
int size;//size用于记录有效数据个数
int capacity;//capacity用于管理空间
}SL;
下面是在定义时需要注意的一些细节:
定义好了顺序表之后我们就来说说增,删,查,改的实现。
增(插入数据):头插,尾插,在指定位置前后插入数据。
删(删除数据):头删,尾删,在指定位置前后删除数据。
查(查找数据):查找指定位置的数据。
改(修改数据):修改指定位置的数据。
在进行这些操作之前我们要对顺序表进行初始化。
//初始化结构体
void SLinit(SL*ps);
//初始化
void SLinit(SL* ps)
{
ps->arr = NULL;
ps->size = ps->capacity = 0;
}
1,增
在插入数据之前我们首先要检查一下,顺序表的空间是否足够不够的话就扩容。
只要是设计插入,就要考虑空间是否足够的问题 所以我们将检查空间是否够封装成一个函数如下:
//判断内存是否足够 封装成一个函数
void SLCheckCapacity(SL*ps);
//检查空间是否足够
void SLCheckCapacity(SL* ps)
{
if (ps == NULL)
{
perror("init fial!");
return;
}
//判断空间是否足够 空间不足需要扩容 扩容需要定义一个新的容量 因为capacity总是指向顺序表最后一个元素的后面
if (ps->size == ps->capacity)
{
int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
//创建一个临时指针指向扩展空间的首地址
SLtype* tmp = (SLtype*)realloc(ps->arr, newcapacity * sizeof(SLtype));//扩容的是新空间
//判断是否开辟成功
if (tmp == NULL)
{
perror("realloc fail!");
return;
}
//走到这开辟成功
ps->arr = tmp;//将新开辟空间的地址给arr 让arr指向新开辟的空间
ps->capacity = newcapacity;//更新容量
}
}
1,头插
具体的代码如下:
头文件中:
//头插
void SLPushFornt(SL* ps, SLtype x);
实现文件中:
//头插
void SLPushFornt(SL* ps, SLtype x)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return ;
}
//assert(ps);
//如果内存不足
SLCheckCapacity(ps);
//如果内存足够
int i = 0;
for (i = ps->size;i > 0;i--)
{
ps->arr[i] = ps->arr[i-1];
}
//程序走到这说明 ps->arr[0]的位置已经被空出来了
ps->arr[0] = x;
++ps->size;
}
2,尾插
头文件中:
//尾插
void SLPushBack(SL* ps, SLtype x);
实现文件中:
//尾插
void SLPushBack(SL* ps, SLtype x)
{
if (ps == NULL)
{
perror("init fial!");
return;
}
//判断空间是否足够 空间不足需要扩容 扩容需要定义一个新的容量 因为capacity总是指向顺序表最后一个元素的后面
if (ps->size == ps->capacity)
{
int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
//创建一个临时指针指向扩展空间的首地址
SLtype* tmp = (SLtype*)realloc(ps->arr, newcapacity * sizeof(SLtype));//扩容的是新空间
//判断是否开辟成功
if (tmp == NULL)
{
perror("realloc fail!");
return ;
}
//走到这开辟成功
ps->arr = tmp;//将新开辟空间的地址给arr 让arr指向新开辟的空间
ps->capacity = newcapacity;//更新容量
}
//空间足够的情况下
ps->arr[ps->size++] = x;
//等价于 ps->arr[ps->size]=x;size++ 这两段代码
}
2,在指定位置前插入数据
//在指定位置之前插入数据
void SLInsert(SL* ps, SLtype pos, SLtype x);
//在指定位置插入数据
void SLInsert(SL* ps, SLtype pos, SLtype x)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return ;
}
//assert(ps);
// 如果内存不足
SLCheckCapacity(ps);
//如果空间足够
for (int i = ps ->size;i > pos;i--)
{
ps->arr[i] = ps->arr[i - 1];
}
ps->arr[pos] = x;//将空出来的位置给x
ps->size = ps->size++;//插入数据 size要++ 即有效数据要++
}
2,删
1,头删
//头删
void SLPopFornt(SL* ps);
//头删
void SLPopFornt(SL* ps)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return ;
}
//assert(ps);
for (int i = 0;i < ps->size;i++)
{
ps->arr[i] = ps->arr[i + 1];
}
ps->size = ps->size--;
}
2,尾删
//尾删
void SLPopBack(SL* ps);
//尾删
void SLPopBack(SL* ps)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return ;
}
//assert(ps);
ps->size = --ps->size;
}
3,删除pos位置(指定位置)的数据
//删除pos位置的数据
void SLErase(SL* ps, SLtype pos, SLtype x);
/删除pos位置的数据
void SLErase(SL* ps, SLtype pos, SLtype x)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return;
}
//assert(ps);
// 如果内存不足
SLCheckCapacity(ps);
for (int i = pos;i < ps->size;i++)
{
ps->arr[i] = ps->arr[i + 1];
}
ps->size = ps->size--;
}
3,查
查找的思路很简单,就是遍历循序表看能否找到我们给定的数据。
//查找
int SLFind(SL* ps, SLtype x);
//查找
int SLFind(SL* ps, SLtype x)
{
//判断ps是否为空指针NULL
if (ps == NULL)
{
perror("fail!");
return;
}
//assert(ps);
//遍历数组查找
for (int i = 0;i < ps->size;i++)
{
if (ps->arr[i] == x)
{
//进来说明找到了
return i;
}
}
//到这说明没找到
return -1;
}
上面函数的返回值之所以是int是为了在测试文件中使用该函数的返回值去判断是否找到了。
4,改
修改的话很简单,根据从查找的方法我们找到指定位置后就可以直接修改数据。代码很简单这里就不再展示。
5,销毁
我们在C语言内存函数中说过,在堆区申请的空间不使用时要释放掉避免造成空间的浪费,动态顺序表是使用malloc开辟的所以不使用时要释放掉。
//销毁
void Destroy(SL* ps);
//销毁
void Destroy(SL* ps)
{
assert(ps);
if (ps->arr)//如果ps—>的arr不为空指针
{
free(ps->arr);//则释放
}
ps->arr = NULL;//释放完后arr变成了野指针 及时将其置为空指针NULL
ps->size = ps->capacity = 0;//释放内存以后顺序表将不复存在 所以将有效数据个数 容量都置为0
}
二,顺序表的总结和问题的思考
时间复杂度:中间/头部的插⼊删除,时间复杂度为O(N)
关于增容的问题:增容需要申请新空间,拷⻉数据,释放旧空间。会有不小的消耗。
增容的注意事项:增容⼀般是呈2倍的增长,势必会有⼀定的空间浪费。例如当前容量 为100,满了以后增容到200,
我们再继续插⼊了5个数据,后⾯没有数据插入了,那么就浪费了95个数据空间。
为了解决以上空间浪费,和增容和时间复杂度优化的问题我们就引出了链表。
三,链表
1,什么是链表?
概念:链表是⼀种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。
光从概念上去看链表还是不好理解,举个例子:链表就是像是一个火车由一个火车头连接着很多节车厢;每节车厢之间都是独立的;且增加或减少车厢对其他的车厢都没有影响。
通过上图我们还发现链表与顺序不一样的点就是链表元素之间是指向关系,而顺序表底层是数组,数组是一块连续的空间。而链表就不一样了,链表在逻辑上是线性的也就是我们人为将他看作是连续的;但在物理空间上链表就是不连续的了。
2,什么是结点?
上图我们看到了链表的结构,链表是由头和结点(车厢)组成的,那么什么是结点呢?
结点就是我们人为的在操作系统的堆区开辟的一块空间,这块空间有连个组成部分一是保存的数据,二是保存下一个结点的地址(指针变量)。
有了上面的认识我们就可以给出结点对应的代码:
typedef struct SListNode //加上typdef修改名字 可以简化代码
{
//结点的两个组成部分 1,保存的数据 2,保存下一个结点的地址
int data; //结点数据
struct SListNode* next; //指针变量⽤保存下⼀个结点的地址
}SL;//将struct ListNode 改名为SL
对于结点(结构体)的解释:
当我们想要保存⼀个整型数据时,实际是向操作系统申请了⼀块内存,这个内存不仅要保存整型数据,也需要保存下⼀个结点的地址(当下⼀个结点为空时保存的地址为空)。
当我们想要从第⼀个结点走到最后一个结点时,只需要在当前结点拿上下一个结点的地址就可以了。
而头结点从图上看就是只存储下一个结点的地址,不存储数据的一个结构体。
从上面的结构我们可以看出链表与顺序表有明显的差异了,顺序表底层使用的是数组;链表使用的是指针。
有了上面的理解,如果给定一个链表让我们打印我们如何实现呢?
//打印函数
void STLPrint(STLNode* phead)
{
STLNode* pcur = phead;//这里重复定义一个pcur是为了避免phead的值发生该变 便于找到链表的头(火车头)
while (pcur)
{
printf("%d -> ", pcur->date);
pcur = pcur->next;//打印完第一个结构体的date数据 就让pcur指向下一个结构体
}
//走到这里说明pcur走到了尽头 此时pcur存的是NULL空指针
printf("NULL\n");
}
3,链表的分类
链表的结构⾮常多样,以下情况组合起来就有8种(2 x 2 x 2)链表结构:
虽然链表有这么多结构但是最常用的有两种单链表(不带头单向不循环链表)和双向链表(双向带头循环链表)。
我们先从单链表开始讲起,下面我们就来讲讲怎么实现单链表:
4,单链表的实现
单链表与顺序表一样,也能实现对数据的增,删,查,改等功能,要实现这些功能我们首先要创建一个链表,在我们定义好结点(结构体)后我们就可以人为的往里面放数据和地址,如下图:
先定义好结点:
//定义链表的结构
typedef int STLDateType;
typedef struct SListNode
{
STLDateType date;
struct SListNode* next; //指向下一个结构体的指针
}STLNode;
创建好了链表以后我们就可以来实现增,删,查,改等功能了。
1,增
在增之前我们肯定要考虑空间问题,由于单链表是动态申请内存的,不会浪费空间需要增加结点的时候才开辟空间,所以我们要设计一个动态申请内存的函数,方便在增加数据的函数中调用。
//定义一个SLTLbuyNode函数用于动态申请内存
STLNode* STLbuyNode(STLDateType x)
{
STLNode* NewNode = (STLNode*)malloc(sizeof(STLNode));//开辟一个结点大小的空间 结点->结构体
if (NewNode == NULL)
{
perror("malloc fail!");
exit(1);
}
//走到这说明开辟结点成功了 将新结点的成员初始化
NewNode->date = x;//x为我们想要传入的值
NewNode->next = NULL;//由于不知道后面还有没有结构体 所以默认置为空指针
return NewNode;//返回新创建好的结点的地址
}
有了这个函数我们就可以开始增加数据了:
1,头插
在.H文件中
//头插
void STLPushFront(STLNode** pphead, STLDateType x);
在.c文件中
//头插
void STLPushFront(STLNode** pphead, STLDateType x)
{
assert(pphead);
//如果链表为空 则改变链表的值
STLNode*NewNode= STLbuyNode(x);
NewNode->next = *pphead;//这是将新的结点与老的第一个结点相连接
*pphead = NewNode;//相连之后让*phead走到前面指向刚创建好的新结点 相当于让pphead往前走
}
2,尾插
尾插这里需要注意到底是传值还是传地址,其实大家在目前阶段可以直接记住只要形参的改变不影响实参就传值,否则就传地址。
//尾插
void STLPushBack(STLNode** pphead, STLDateType x);
//尾插
void STLPushBack(STLNode** pphead, STLDateType x)//由于传进来的plist是一级指针 所以要用二级指针来接收
{
//注意phead传进来的是链表的头节点
STLNode* NewNode = STLbuyNode(x);//直接调用STLbuyNode函数创建一个新的结点
//链表为空的情况 此时头结点就应该是我们插入的新结点
if (*pphead == NULL)
{
//要想该百年实从参还得是使用指针 传址调用
*pphead = NewNode;
//这种 变化也是无效的变化 因为我们的目的是要让我们传进来的plist 指向我们创建好的这个新结点 作为第一个结点
//phead = NewNode;
//下面这种方法是错误的 空链表内什么都没有 哪里来的next成员
//phead->next = NewNode;
}
else
{
//链表不为空的情况
STLNode* ptail = *pphead;//定义一个找尾的指针 让他找到最后一个结点的最后一个成员
while (ptail->next != NULL)
{
ptail = ptail->next;
}
//走到这说明 ptail已经为最后一个结点的next成员了 要将ptail与新结点链接起来
ptail->next = NewNode;//NewNode里边存了新结点的地址
}
}
3,在指定位置之前插入数据
//在指定位置前插入数据
void STLInsert(STLNode** pphead, STLNode*Pos,STLDateType x);
//在指定位置之前插入数据
void STLInsert(STLNode** pphead, STLNode* pos, STLDateType x)
{
assert(pphead && pos && *pphead);
//当pos指向第一个结点 就是头插
if (pos == *pphead)
{
STLPushFront(*pphead,x);
}
//开辟一个新结点
STLNode* NewNode = STLbuyNode(x);
STLNode* prev = *pphead;//定义一个prev来遍历链表
//找到pos的前一个结点
while (prev->next != pos)//注意这里只是判断
{
prev = prev->next;
}
//走到这里 此时prev->next成员指向的是pos位置之后的结点的地址
//走到这里prev->next==pos 注意这里是prev的成员指向了pos位置 说明prev已经走到了pos的前一个结点的位置
prev->next=NewNode;
NewNode->next=pos;
}
4,在指定位置之后插入数据
//在指定位置之后插入数据
void STLInsertAfter(STLNode* pos, STLDateType x);
//在指定位置之后插入数据
void STLInsertAfter(STLNode* pos, STLDateType x)
{
assert(pos);
STLNode* NewNode = STLbuyNode(x);
//pos NewNode pos->next
NewNode->next = pos->next;
pos->next = NewNode;
}
1,删
1,头删
//头删
void STLPopFront(STLNode** pphead);
//头删
void STLPopFront(STLNode** pphead)
{
assert(pphead && *pphead);
STLNode* next = (*pphead)->next;//定义一个next指针来保存*pphead的next成员的值
free(*pphead);
*pphead = next;
}
2,尾删
//尾删
void STLPopBack(STLNode** pphead);
//尾删
void STLPopBack(STLNode** pphead)
{
assert(pphead && *pphead);//
//只有一个结点的情况
if ((*pphead)->next == NULL)
{
free(*pphead);
*pphead = NULL;
}
//有多个结点的情况
else
{
STLNode* ptail = *pphead;
STLNode* prev = NULL;//定义一个prev跟在ptail的后面
while (ptail->next)
{
prev = ptail;
ptail = ptail->next;
}
//循环结束时,ptail为NULL,prev指向最后一个结点
prev->next = NULL;//将倒数第二个结点的next指针置为空
free(ptail);//释放原本最后一个结点(结构体)的空间
ptail = NULL;//最后一个结点被释放了 避免野指针所以置为空
}
}
3,删除pos位置之前的结点
//删除pos位置的结点
void STLErase(STLNode** pphead, STLNode* pos);
//删除pos位置的结点
void STLErase(STLNode** pphead, STLNode* pos)
{
assert(pphead && pos);
//判断prev是不是指向头结点
if (pos == *pphead)
{
STLPopFront(pphead);
}
else
{
STLNode* prev = *pphead;
while (prev->next != pos)//注意循环条件
{
prev = prev->next;
}
//走到这 prev就已经找到pos前一个位置了
prev->next = pos->next;//将prev位置处的结点与pos后的结点相连
free(pos);
pos = NULL;//防止pos为野指针
}
}
4,删除pos位置之后的结点
//删除pos之后的结点
void SLTEraseAfter(SLTNode* pos);
//删除pos之后的结点
void SLTEraseAfter(SLTNode* pos)
{
assert(pos && pos->next);
//pos del del->next
SLTNode* del = pos->next;
pos->next = del->next;
free(del);
del = NULL;
}
3,查
//查找数据
STLNode* STLFind(STLNode* phead, STLDateType x);
//查找数据
STLNode* STLFind(STLNode* phead, STLDateType x)
{
STLNode* pcur = phead;
while (pcur->next != NULL)//让pcur去遍历链表
{
if (pcur->date == x)
{
return pcur;
}
pcur = pcur->next;//让pcur偏移
}
return NULL;
}
4 ,链表的销毁
链表是我们人为开辟的空间,所以为了避免空间的浪费在我们不使用链表时就手动将他销毁。
//销毁链表
void SListDestroy(SLTNode** pphead);
//销毁链表
void SListDestroy(SLTNode** pphead)
{
SLTNode* pcur = *pphead;
while (pcur)
{
SLTNode* next = pcur->next;
free(pcur);
pcur = next;
}
*pphead = NULL;
}
以上就是所有单链表的实现了,由于篇幅有限剩下的双链表我们下期再介绍。
以上就是本章的全部内容啦!
最后感谢能够看到这里的读者,如果我的文章能够帮到你那我甚是荣幸,文章有任何问题都欢迎指出!制作不易还望给一个免费的三连,你们的支持就是我最大的动力!