哲♂学家带你深♂入了解动态顺序表

前言:

最近本哲♂学家学习了顺序表,下面我给大家分享一下关于顺序表的知识。

一、什么是顺序表

  • 顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组
    上完成数据的增删查改。

  • 顺序表:可动态增长的数组,要求数据是连续存储的。

 二、顺序表

1、顺序表的定义

typedef int SLDataType; //类型重命名,后续要存储其它类型时方便更改typedef struct SeqList
{SLDataType* arr;    //指向动态开辟的数组size_t size;      //有效数据个数size_t capacity;  //容量大小
}SL;

注意将数组定义为数组指针:如果定义为数组则属性表空间太大,造成空间浪费。

2、顺序表的文件

  • SeqList.h(顺序表的类型定义、接口函数声明、引用的头文件)
  • SeqList.c(顺序表接口函数的实现)
  • Test.c(主函数、测试顺序表各个接口功能)

 3、头文件

#pragma once//防止头文件二次引用
typedef int SLDataType;//方便后续改存储类型
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef struct seqlist
{SLDataType* arr;//存放数据的数组int size;//有效元素个数int capacity;//空间的大小
}SL;
// 初始化和销毁
void SLInit(SL * ps);
void SLDestroy(SL* ps);
void SLPrint(SL* ps);
//扩容
void SLCheckCapacity(SL* ps);//头部插入删除 / 尾部插入删除
void SLPushBack(SL* ps, SLDataType x);
void SLPopBack(SL* ps);
void SLPushFront(SL* ps, SLDataType x);
void SLPopFront(SL* ps);//指定位置之前插入/删除数据
void SLInsert(SL* ps, int pos, SLDataType x);
void SLErase(SL* ps, int pos);
//找到指定位置的数据
int SLFind(SL* ps, SLDataType x);

三、函数的实现

1、顺序表的初始化与销毁

//初始化
void SLInit(SL* ps)
{ps->arr = NULL;ps->capacity = 0;ps->size = 0;
}
//销毁
void SLDestroy(SL* ps)
{assert(ps->arr);//防止对空指针释放空间if (ps->arr != NULL)//或者直接写ps->arr{free(ps->arr);}ps->arr = NULL;ps->capacity = ps->size = 0;
}

2、顺序表的打印

为了方便后续观察代码的正确性,我们可以写出打印函数

//打印
void SLPrint(SL* ps)
{assert(ps->arr);//防止为空for (int i = 0; i < ps->size; i++){printf("%d", ps->arr[i]);}printf("\n");
}

3、扩容

注意我们扩容空间的时候不能改变size的值,因为我们设定size为有效元素个数。

void SLCheckCapacity(SL* ps)//如果空间不够就经行扩容
{if (ps->capacity == ps->size)//这样说明空间不够{int newcapacity = (ps->capacity == 0 ? 4 : 2 * ps->capacity);//是0则新的空间大小为4,不是则为2倍原来空间大小SLDataType* tmp = (SLDataType*) realloc(ps->arr, newcapacity * sizeof(SLDataType));if (tmp == NULL)//如果没有开辟成功{perror("realloc fail!");exit(1);//直接退出程序,不再继续执行}//如果开辟成功ps->arr = tmp;ps->capacity = newcapacity;tmp = NULL;//完事后将创建的临时指针放置为空}
}

事实上这一步是在为后面插入数据做准备。

4、尾插和尾删

1、尾插:

void SLPushBack(SL* ps, SLDataType x)
{SLCheckCapacity(ps);//检查空间是否够用assert(ps->arr);ps->arr[ps->size] = x;ps->size++;//或者直接写出ps->arr[ps->size++] = x;
}

2、尾删

这里可能会想到将要删除的数据置为0,实际上这样还是会占用开辟的空间,那不如我们不用将删除数据置0,直接将有效数据个数减一即可。(同时只有当数据表数据个数不为0才可以删除)

void SLPopBack(SL* ps)
{assert(ps->arr);assert(ps->size);//判断数据是否为0ps->size--;
}

测试后发现是对的。

5、头插和头删

1、头插

头插是将数据放在头部,所以后面的数据要整体向后移动,所以我们可以简单画个图

如图我们可以发现是将size-1的数据移动到size的位置所以代码可以这么写:

void SLPushFront(SL* ps, SLDataType x)
{SLCheckCapacity(ps);//检查空间是否够用assert(ps->arr);for (int i = 0; i < ps->size; i++)//移到后面。{ps->arr[ps->size - i] = ps->arr[ps->size - 1 - i];}//或者这么写//for (int i = ps->size; i > 0; i--)//{//	ps->arr[i] = ps->arr[i - 1];//arr[1] = arr[0]//}ps->arr[0] = x;ps->size++;
}

2、头删

头删只不过是头插的逆过程我们直接将后面的数据向前移动不用管第一个数据即可:

void SLPopFront(SL* ps)
{assert(ps);assert(ps->size);//数据整体往前挪动一位for (int i = 0; i < ps->size - 1; i++){ps->arr[i] = ps->arr[i + 1]; //arr[size-2] = arr[size-1]}ps->size--;
}

测试发现没有任何问题

6、指定位置之前插入和删除数据

1、插入 

指定位置插入实际上是头插的一种特殊形式(有个偷懒的办法最后写出的条件带入size和0看是否与上面的形式相同即可):

//指定位置插
void SLInsert(SL* ps, int pos, SLDataType x)
{SLCheckCapacity(ps);//检查空间是否够用assert(ps->arr);for (int i = ps->size; i > pos; i--)//后面的向后移动{ps->arr[i] = ps->arr[i - 1];}ps->arr[pos] = x;ps->size++;
}

2、删除

删除同理:

void SLErase(SL* ps, int pos)
{assert(ps);  //断言assert(ps->size);  //顺序表不能为空assert(pos >= 0 && pos < ps->size);  //检查pos下标的合法性int i = 0;for (i = pos + 1; i < ps->size; i++)  //将pos位置后面的数据依次向前挪动一位{ps->arr[i - 1] = ps->arr[i];}ps->size--;  //有效数据个数-1
}

测试发现没有问题

7、找到指定位置的数据

这个就更简单了,只需要将数组遍历一遍即可。

int SLFind(SL* ps, SLDataType x)
{for (int i = 0; i < ps->size; i++){if (ps->arr[i] == x){return i;//找到返回下标}}return -1;//找不到返回无效值
}

总结

顺序表是比数组更加灵活的存储方式,也是我们接触的第一个数据结构,熟练掌握是十分重要的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/296957.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Rust线程间通信通讯channel的理解和使用

Channel允许在Rust中创建一个消息传递渠道&#xff0c;它返回一个元组结构体&#xff0c;其中包含发送和接收端。发送端用于向通道发送数据&#xff0c;而接收端则用于从通道接收数据。不能使用可变变量的方式&#xff0c;线程外面修改了可变变量的值&#xff0c;线程里面是拿不…

水果销售(源码+文档)

水果销售管理系统&#xff08;小程序、ios、安卓都可部署&#xff09; 文件包含内容程序简要说明含有功能项目截图客户端添加地址首页商品详细意见反馈待发货商品分类我的代付款我的地址搜索防骗指南资料修改登录注册 后端管理分类管理反馈管理订单管理商品管理用户管理 文件包…

OpenHarmony实战:轻量级系统之配置其他子系统

除上述子系统之外&#xff0c;还有一些必要但是无需进行移植的子系统。如&#xff1a;分布式任务调度子系统、DFX子系统。 这些子系统添加方式比较简单&#xff0c;在“vendor/MyVendorCompany/MyProduct/config.json”文件中进行如下配置即可&#xff1a; {"subsystem&…

vulhub中Apache Solr 远程命令执行漏洞复现(CVE-2019-0193)

Apache Solr 是一个开源的搜索服务器。Solr 使用 Java 语言开发&#xff0c;主要基于 HTTP 和 Apache Lucene 实现。此次漏洞出现在Apache Solr的DataImportHandler&#xff0c;该模块是一个可选但常用的模块&#xff0c;用于从数据库和其他源中提取数据。它具有一个功能&#…

[RK3128_LINUX5.1] 关于 RetroArch 使用

问题描述 查看文档 docs\cn\Linux\ApplicationNote\Rockchip_Use_Guide_Linux_RetroArch_CN.pdf&#xff0c;描述为实验 make menuconfig 后勾选选项 Libretro cores and retroarch -> retroarch 但是SDK中并没有这个选项 解决方案&#xff1a; 目前发布的buildroot SDK…

7 X 24h智能安全运维再升级!Fortinet 全面集成全新 FortiGuard SOCaaS

数字化时代网络安全威胁层出不穷&#xff0c;网络犯罪分子的狡诈攻击手段不断翻新&#xff0c;传统安全防御手段亟需进化。更为棘手的是&#xff0c;网络安全专业人才的匮乏&#xff0c;让众多企业陷入安全运营的困境。为了有效应对这一挑战&#xff0c;Fortinet全新推出FortiG…

使用PostgreSQL中的隐式转换解决,MybatisPlus插入数据库时的类型不一致的问题

使用PostgreSQL中的隐式转换解决,MybatisPlus插入数据库时的类型不一致的问题 问题描述 鄙人在使用 MybatisPlus插件开发一个SpringBoot项目时, 遇到数据库中employee表与Java实体对象中某个属性的类型不一致, 导致插入数据库失败. 具体问题截图如下: 具体原因在于, Java实体…

泛微E-Mobile /client.do 命令执行漏洞复现

0x01 产品简介 泛微e-Mobile移动管理平台是一款由泛微软件开发的企业移动办公解决方案。它提供了一系列功能和工具,使企业员工能够通过移动设备随时随地进行办公和协作。 0x02 漏洞概述 泛微E-Mobile存在命令执行漏洞,攻击者可以通过/client.do执行任意命令执行,从而获得…

Golang学习系列1-pprof性能调优

1. pprof 简述 一位亦师亦友的话让我记忆犹新&#xff0c;他说“学习一个新事务&#xff0c;应该从三个方面入手what,why,how;且三者的重要程度应该是递减”。所以在本文的第一部分先叙述下pprof的what & why。 1.1 What&#xff1f; pprof是golang自身提供的一种性能分…

C++初学者:优雅创建第一个窗口

我想学习C做一些实用的程序&#xff0c;但是我不想在软件界面上花太多的时间&#xff0c;可是每每就是界面影响我的思绪。 今天学习C类的包装知识&#xff0c;终于整出了一个我的界面类&#xff0c;虽然封装水平很弱&#xff0c; 这次就用这个类&#xff0c;写了自己工作上常用…

高阶DS---AVL树详解(每步配图)

目录 前言&#xff1a; AVL树的概念: AVL树节点的定义&#xff1a; AVL树的插入&#xff08;重点&#xff09; AVL树的旋转&#xff1a; &#xff08;1&#xff09;新节点插入较高左子树的左侧---右单旋 &#xff08;2&#xff09;新节点插入较高右子树的右侧---左单旋 …

续二叉搜索树递归玩法

文章目录 一、插入递归二、寻找递归&#xff08;非常简单&#xff0c;走流程就行&#xff09;三、插入递归&#xff08;理解起来比较麻烦&#xff09; 先赞后看&#xff0c;养成习惯&#xff01;&#xff01;&#xff01;^ _ ^<3 ❤️ ❤️ ❤️ 码字不易&#xff0c;大家的…

【基于HTML5的网页设计及应用】——-正则表达式.

&#x1f383;个人专栏&#xff1a; &#x1f42c; 算法设计与分析&#xff1a;算法设计与分析_IT闫的博客-CSDN博客 &#x1f433;Java基础&#xff1a;Java基础_IT闫的博客-CSDN博客 &#x1f40b;c语言&#xff1a;c语言_IT闫的博客-CSDN博客 &#x1f41f;MySQL&#xff1a…

ObjectiveC-09-OOP面向对象程序设计-继承

继承是所有高级语言都实现的一种特征&#xff0c;比如java、python甚至现在的js也有类似继承的写法。在表层目的是减少重复代码&#xff0c;深层目的是为了匹配业务与程序间的映射关系。 先来看下OOP设计思起中的继承的关系表示 再结合实例来看下完整的关系表示 一个简单的例…

10 年跟踪 Hacker News 招聘贴,解读科技行业变迁

Hackers News (HN) 是国外程序员最喜欢逛的论坛。能登上首页的帖子类似于上了新浪微博。因为其巨大的程序员访问量&#xff0c;因此也成为了公司招聘的渠道。久而久之 HN 招聘帖还形成了专门的标题格式 Ask HN: Who is hiring? 正好有人通过 Ask HN 来分析技术趋势&#xff0c…

【Blockchain】区块链浏览器 | 以太坊Etherscan比特币Blockchain门罗币Monero

区块链浏览器概述 区块链浏览器是一种软件,它使用API(应用程序编程接口)和区块链节点从区块链中提取各种数据&#xff0c;然后使用数据库来排列搜索到的数据&#xff0c;并以可搜索的格式将数据呈现给用户。 用户的输入是资源管理器上的可搜索项&#xff0c;然后通过数据库上…

《QT实用小工具·十》本地存储空间大小控件

1、概述 源码放在文章末尾 本地存储空间大小控件&#xff0c;反应电脑存储情况&#xff1a; 可自动加载本地存储设备的总容量/已用容量。进度条显示已用容量。支持所有操作系统。增加U盘或者SD卡到达信号。 下面是demo演示&#xff1a; 项目部分代码如下&#xff1a; #if…

数据库管理工具 DBeaverUE for Mac激活版

DBeaverUE for Mac是一款功能强大且易于使用的数据库管理工具&#xff0c;专为Mac用户设计。它支持多种数据库类型&#xff0c;如MySQL、PostgreSQL、Oracle等&#xff0c;使得用户可以轻松管理和操作各种数据库。 软件下载&#xff1a;DBeaverUE for Mac激活版下载 DBeaverUE …

【C语言】翻译环境与运行环境

一、前言 在我们学习C语言的时候&#xff0c;第一个接触的程序就是&#xff1a;在屏幕上打印” hello word! “&#xff0c;可当时的我们却未去深入的理解与感悟&#xff0c;一个程序代码是如何运行的&#xff1b;而这一期的博客&#xff0c;则是带着我们&#xff0c;通过C代码…

STM32的定时器中断Cubemx

STM32的定时器中断Cubemx 0.定时器简介1.配置时钟2.配置定时器3.创建工程4.补充源码 0.定时器简介 基本定时器功能&#xff1a; 16位向上、向下、向上/下自动装载计数器16位可编程(可以实时修改)预分频器&#xff0c;计数器时钟频率的分频系数为1&#xff5e;65535之间的任意…