顺序表和链表
顺序表和链表
线性表 ( linear list ) 是 n 个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使
用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串 ...
线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,
线性表在物理上存储时,通常以数组和链式结构的形式存储
2.顺序表
顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存 储。在数组上完成数据的增删查改
2.1顺序表的分类
顺序表一般可以分为: 1. 静态顺序表:使用定长数组存储元素。
2. 动态顺序表:使用动态开辟的数组存储。
2.2 接口实现
静态顺序表只适用于确定知道需要存多少数据的场景。静态顺序表的定长数组导致定大了,空 间开多了浪费,开少了不够用。所以现实中基本都是使用动态顺序表,根据需要动态的分配空间 大小,所以下面我们实现动态顺序表。
typedef int SLDataType; // 顺序表的动态存储 typedef struct SeqList { SLDataType*arr;//动态开辟的数组 szie_t size;//有效数据大小 size_t capacity;//有效的容量 } // 基本增删查改接口 // 顺序表初始化 void SeqListInit(SeqList* psl); // 检查空间,如果满了,进行增容 void CheckCapacity(SeqList* psl); // 顺序表尾插 void SeqListPushBack(SeqList* psl, SLDataType x); // 顺序表尾删 void SeqListPopBack(SeqList* psl); // 顺序表头插 void SeqListPushFront(SeqList* psl, SLDataType x); // 顺序表头删 void SeqListPopFront(SeqList* psl); // 顺序表查 int SeqListFind(SeqList* psl, SLDataType x); // 顺序表在pos位置插入x void SeqListInsert(SeqList* psl, size_t pos, SLDataType x); // 顺序表删除pos位置的值 void SeqListErase(SeqList* psl, size_t pos); // 顺序表销毁 void SeqListDestory(SeqList* psl); // 顺序表打印 void SeqListPrint(SeqList* psl);
2.顺序表的问题及思考
问题: 1. 中间/头部的插入删除,时间复杂度为O() 2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。 . 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到 200,我们再继续插入了5个数据,后面没有数据插入了,那么就浪费了95个数据空间。 思考:如何解决以上问题呢?下面给出了链表的结构来看看。
.1 链表的概念及结构
概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表 中的指针链接次序实现的 。
现实中 数据结构中
.2 链表的分类
实际中链表的结构非常多样,以下情况组合起来就有8种链表结构
1. 单向或者双向
2. 带头或者不带头
. 循环或者非循环
虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:
1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结 构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。 2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都 是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带 来很多优势,实现反而简单了,后面我们代码实现了就知道了。
. 链表的实现
// 1、无头+单向+非循环链表增删查改实现 typedef int SLTDateType; typedef struct SListode { SLTDateType data; struct SListode* next; }SListode; // 动态申请一个结点 SListode* BuySListode(SLTDateType x); // 单链表打印 void SListPrint(SListode* plist); // 单链表尾插 void SListPushBack(SListode** pplist, SLTDateType x); // 单链表的头插 void SListPushFront(SListode** pplist, SLTDateType x); // 单链表的尾删 void SListPopBack(SListode** pplist); // 单链表头删 void SListPopFront(SListode** pplist); // 单链表查 SListode* SListFind(SListode* plist, SLTDateType x); // 单链表在pos位置之后插入x // 分析思考为什么不在pos位置之前插入? void SListInsertAfter(SListode* pos, SLTDateType x); // 单链表删除pos位置之后的值 // 分析思考为什么不删除pos位置? void SListEraseAfter(SListode* pos);
.4 双向链表的实现
// 2、带头+双向+循环链表增删查改实现 typedef int LTDataType; typedef struct Listode { LTDataType _data; struct Listode* next; struct Listode* prev; }Listode; // 创建返回链表的头结点. Listode* ListCreate(); // 双向链表销毁 void ListDestory(Listode* plist); // 双向链表打印 void ListPrint(Listode* plist); // 双向链表尾插 void ListPushBack(Listode* plist, LTDataType x); // 双向链表尾删 void ListPopBack(Listode* plist); // 双向链表头插 void ListPushFront(Listode* plist, LTDataType x); // 双向链表头删 void ListPopFront(Listode* plist); // 双向链表查 Listode* ListFind(Listode* plist, LTDataType x); // 双向链表在pos的前面进行插入 void ListInsert(Listode* pos, LTDataType x); // 双向链表删除pos位置的结点 void ListErase(Listode* pos);
不同点 | 顺序表 | 链表 |
---|---|---|
存储空间上 | 物理上一定连续 | 逻辑上连续,但物理上不一定 连续 |
随机访问 | 支持O(1) | 不支持:O() |
任意位置插入或者删除 元素 | 可能需要搬移元素,效率低 O() | 只需修改指针指向 |
插入 | 动态顺序表,空间不够时需要 扩容 | 没有容量的概念 |
应用场景 | 元素高效存储+频繁访问 | 任意位置插入和删除频繁 |
缓存利用率 | 高 | 低 |
备注:缓存利用率参考存储体系结构 以及 局部原理性。
#感谢您对电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格的认可,转载请说明来源于"电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格
推荐阅读
留言与评论(共有 8 条评论) |
本站网友 阿泰宁 | 1分钟前 发表 |
空 间开多了浪费 | |
本站网友 股票年线 | 24分钟前 发表 |
size_t pos | |
本站网友 上海治疗早泄的医院 | 20分钟前 发表 |
LTDataType x); // 双向链表尾删 void ListPopBack(Listode* plist); // 双向链表头插 void ListPushFront(Listode* plist | |
本站网友 新天地ume | 15分钟前 发表 |
根据需要动态的分配空间 大小 | |
本站网友 大鹏背景 | 27分钟前 发表 |
效率低 O()只需修改指针指向插入动态顺序表 | |
本站网友 香皂品牌 | 22分钟前 发表 |
SLDataType x); // 顺序表尾删 void SeqListPopBack(SeqList* psl); // 顺序表头插 void SeqListPushFront(SeqList* psl | |
本站网友 维生素b2的作用 | 13分钟前 发表 |
分享自作者个人站点/博客 |