数据结构(C):玩转顺序表

🍺0.前言

🎷1.线性表

🎸2.顺序表

📀动态顺序表的实现

💿初始化

💿检查容量是否满了,进行扩容

💿插入:头插和尾插

💿删除:头删和尾删

💿查找

💿销毁

💿打印

💿在某处插入和删除

💎6.结束语


🍺0.前言

        言C之言,聊C之识,以C会友,共向远方。各位博友的各位你们好啊,这里是持续分享数据结构知识的小赵同学,今天要分享的数据结构知识是顺序表,在这一章,小赵将会向大家展开聊聊顺序表。✊

🎷1.线性表

线性表 linear list n 个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使 用的数据结构, 常见的线性表:顺序表、链表、栈、队列、字符串...
线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的, 线性表在物理上存储时,通常以数组和链式结构的形式存储。

 也就是说我们今天要谈的顺序表就属于我们的线性表。

🎸2.顺序表

顺序表是用一段 物理地址连续 的存储单元依次存储数据元素的线性结构,一般情况下采用数组存 储。在数组上完成数据的增删查改。
顺序表一般可以分为:
1. 静态顺序表:使用定长数组存储元素。
2.动态顺序表:使用动态开辟的数组存储。

好了下面我们来看看究竟怎么样才是一个顺序表,其实很简单,小赵在这里给各位画个图

怎么样它的样子是不是非常的眼熟,没错我们的线性表其实就是我们的数组。那为什么有静态和动态之分呢?因为一个可以不断增长自己的内存,一个则是固定容量,那该如何实现呢?下面小赵将会带着大家去实现顺序表。

因为静态顺序表其实和动态的线性表差距不大,而且静态顺序表本身也并不常用(开大了浪费,开小了,不够),所以小赵在这里主要带着大家去实现我们的动态顺序表。

📀动态顺序表的实现

那么动态的顺序表该如何实现呢?在这里为了方便我们在后续知道什么时候该进行扩大我们的数组,我们就用我们的结构体去实现我们的线性动态表。

typedef int SeqDataType;
struct Seqlist
{SeqDataType* a;//指向动态开辟的数组int size;//顺序表已存元素个数int capacity;//顺序表容量
};

 那么这样当我们的size 等于我们的capacity的时候我们就可以对我们的数组进行扩容了。

那么对于我们的顺序表,光有这个是不够的,还要具备增删查改,初始化等等功能,下面带着大家一一来实现

💿初始化:

首先是初始化的操作

typedef struct Seqlist SL;
void InitSeq(SL* psl)
{assert(psl);//防止传入空指针psl->a = NULL;psl->capacity = 0;psl->size = 0;
}

💿检查容量是否满了,进行扩容

void SLCheckcapacity(SL* psl)
{assert(psl);if (psl->capacity == psl->size){int newcapacity = (psl->capacity==0?4: psl->capacity *2);//如果原容量为零则返回4,其他返回原容量的两倍SeqDataType *tmp = (SeqDataType*)realloc(psl->a ,sizeof(SeqDataType) *newcapacity);//reallc新数组if (tmp == NULL){perror("malloc failed");return;}psl->a = tmp;psl->capacity = newcapacity;}
}

💿插入:头插和尾插

这里的头插要稍微注意一下,就像我们站个队,忽然要我们在前面给个位子一样,那么我们是不是后面的人都要向后移动,那么如何向后移动呢,如果是从头移动,我们会发现下面的问题。

我们会发现我们前面的数字会盖住后面的数字,那么这里最后的办法就是后面向后移动,给出位子。

然后就可以顺利解决了。代码如下

void SLPushFront(SL* psl, SeqDataType x)
{assert(psl);SLCheckcapacity(psl);//检查容量int n = psl->size-1;while (n--);//将所有元素向后移动{psl->a[n + 1] = psl->a[n];}psl->a[0] = x;psl->size++;
}

 尾插就相对比较容易了。

void SLPushBack(SL* psl, SeqDataType x)
{assert(psl);SLCheckCapacity(psl);psl->a[psl->size++] = x;
}

💿删除:头删和尾删

这里的头删的思路和我们的头插的思路很像,只是这里是头边有空位

void SLPopFront(SL* psl)
{assert(psl);assert(psl->size);//防止元素个数为0int n = 1;while (n < psl->size){psl->a[n-1] = psl->a[n];//把后面的往前移n++;}psl->size--;
}

尾删则相对容易些: 

void SLPopBack(SL* psl)
{assert(psl);assert(psl->size);//防止元素个数为0psl->size--;
}

💿查找:

int SLFind(SL* psl, SeqDataType x)
{assert(psl);int n = 0;while (n<psl->size){if (psl->a[n] == x)//找到了返回对应位置return n;n++;}return -1;//找不到
}

💿销毁:

void SLDestory(SL* psl)
{assert(psl);free(psl->a);psl->a = NULL;psl->capacity = 0;psl->size = 0;
}

💿打印:

void SLPrint(SL* psl)
{assert(psl);int n = 0;while (n < psl->size){printf("%d", psl->a[n]);n++;}
}

💿在某处插入和删除:

在某处的插入和删除对于逻辑的考察是有的,那么我们先看看如何进行插入:

比方说,我要在三这个位置进行插入操作,那么这个时候3,4,5这几个数字是不是要向后移动,那么其就很像我们的头删,都是从尾部开始向后移动,来完成的。

void SLInsert(SL* psl, int pos, SeqDataType x)
{assert(psl);assert(0 <= pos && pos <= psl->size);SLCheckcapacity(psl);int end= psl->size-1;while (end >= pos){psl->a[end + 1] =psl->a[end];end--;}psl->a[pos] = x;psl->size++;
}

接下来是某个位置的删除,类比头删

void SLErase(SL* psl, int pos, SeqDataType x)
{assert(psl);assert(0 <= pos && pos < psl->size);int end = pos;while (end<psl->size-1){psl->a[end] = psl->a[end+1];end++;}psl->size--;
}

 然后我们还可以用这两个代码改我们原本的头尾插,头尾删

void SLPushFront(SL* psl, SeqDataType x)
{SLInsert(psl, 0, x);
}
void SLPushBack(SL* psl, SeqDataType x)
{SLInsert(psl, psl->size, x);
}
void SLPopFront(SL* psl)
{SLErase(psl, 0);
}
void SLPopBack(SL* psl)
{SLErase(psl, psl->size-1);
}

 

💎6.结束语

好了小赵今天的分享就到这里了,如果大家有什么不明白的地方可以在小赵的下方留言哦,同时如果小赵的博客中有什么地方不对也希望得到大家的指点,谢谢各位家人们的支持。你们的支持是小赵创作的动力,加油。

如果觉得文章对你有帮助的话,还请点赞,关注,收藏支持小赵,如有不足还请指点,小赵及时改正,感谢大家支持!!! 

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

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

相关文章

基于PSO优化的PV光伏发电系统simulink建模与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 5.完整工程文件 1.课题概述 基于PSO优化的PV光伏发电系统simulink建模与仿真。其中PSO采用matlab编程实现&#xff0c;通过simulink的函数嵌入模块&#xff0c;将matlab调用进simulink中。 2.系统仿真结…

【强训笔记】day10

NO.1 思路&#xff1a;中心扩展。从i位置开始&#xff0c;从i-1为左边和i1右边进行移动&#xff0c;字符相等就继续移动&#xff0c;直到不等&#xff0c;更新回文串长度&#xff0c;让i为左边&#xff0c;i1右边再移动&#xff0c;同样字符相等就移动&#xff0c;不等就更新长…

五、VGA 叠加图像原理和实现(十字光标)

前言&#xff1a;该案例在VGA项目&#xff1a;联合精简帧双fifosobel算法 实现VGA显示项目的基础上进行改动。 要求&#xff1a;通过串口助手把 198x198 的十字光标图像二进制数据传递给 FPGA 板卡&#xff0c;FPGA 板 卡接收到后存储到 Ram 中用于 VGA 叠加显示。 预期效果展…

为什么电子商务安全是速度和保护之间的平衡行为

微信搜索关注公众号网络研究观&#xff0c;获取更多信息。 电子商务世界是一把双刃剑。虽然它为企业和消费者提供了便利和可访问性&#xff0c;但它也为网络犯罪分子提供了诱人的目标。在这个不断变化的环境中&#xff0c;优先考虑安全不再是一种选择&#xff1b;而是一种选择&…

解决Win10家庭版找不到组策略gpedit.msc的·方法

因为电脑出问题&#xff0c;一开机就会自动开启ie浏览器&#xff0c;所以就想找有没有方法解决&#xff0c;然后就了解到了gpedit.msc的作用以及相关的一些方法&#xff0c;也是为之后也许有人遇到相同的问题有个提供方法的途径。 首先我们直接运行gpedit.msc 是找不到的&…

vivado刷题笔记46

题目&#xff1a; Design a 1-12 counter with the following inputs and outputs: Reset Synchronous active-high reset that forces the counter to 1 Enable Set high for the counter to run Clk Positive edge-triggered clock input Q[3:0] The output of the counter c…

【网络知识】光猫、路由器 和 交换机 的作用和区别?

数字信号&#xff1a;是指自变量是离散的、因变量也是离散的信号&#xff0c;这种信号的自变量用整数表示&#xff0c;因变量用有限数字中的一个数字来表示。在计算机中&#xff0c;数字信号的大小常用有限位的二进制数表示。 模拟信号&#xff1a;模拟信号是指用连续变化的物…

神经网络中的归一化

我们今天介绍一下神经网络中的归一化方法~ 之前学到的机器学习中的归一化是将数据缩放到特定范围内&#xff0c;以消除不同特征之间的量纲和取值范围差异。通过将原始数据缩放到一个特定的范围内&#xff0c;比如[0,1]或者[-1,1]&#xff0c;来消除不同特征之间的量纲和取值范围…

Excel利用数据透视表将二维数据转换为一维数据(便于后面的可视化分析)

一维数据&#xff1a;属性值都不可合并&#xff0c;属性值一般在第一列或第一行。 二维数据&#xff1a;行属性或列属性是可以继续合并的&#xff0c;如下数据中行属性可以合并为【月份】 下面利用数据透视表将二维数据转换为一维数据&#xff1a; 1、在原来的数据上插入数据透…

机器人系统ros2-开发实践05-将静态坐标系广播到 tf2(Python)-定义机器人底座与其传感器或非移动部件之间的关系

发布静态变换对于定义机器人底座与其传感器或非移动部件之间的关系非常有用。例如&#xff0c;最容易推断激光扫描仪中心框架中的激光扫描测量结果。 1. 创建包 首先&#xff0c;我们将创建一个用于本教程和后续教程的包。调用的包learning_tf2_py将依赖于geometry_msgs、pyth…

蛋白质/聚合物防污的机器学习(材料基因组计划)

前言&#xff1a;对于采用机器学习去研究聚合物的防污性能&#xff0c;以及或者其他性质。目前根据我的了解我认为最困难的点有三条&#xff1a; 其一&#xff1a;数据&#xff0c;对于将要训练的数据必须要有三点要求&#xff0c;1.数据要多&#xff0c;也就是大数据&#xff…

设置定位坐标+请按任意键继续

设置定位坐标 目的 在编程和游戏开发中&#xff0c;设置定位坐标的目的是为了确定对象在屏幕或游戏世界中的具体位置。坐标通常由一对数值表示&#xff0c;例如 (x, y)&#xff0c;其中 x 表示水平位置&#xff0c;y 表示垂直位置。设置定位坐标的目的包括&#xff1a; 1. **精…