【数据结构初阶】--- 栈和队列

栈的定义

栈:只允许在一端进行插入或删除的操作
事实上,线性表和链表都可以实现栈,但栈的特点更符合用顺序表实现

  • 顺序表的队尾相当于栈顶,对栈放入数据,相当于顺序表的下标arr[index++] = x,而栈弹出数据时,相当顺序表的下标index–,即可。
  • 相比之下链表就没有这么的符合,入栈时,链表为其创建一个结点进行尾插,出栈时,进行尾删,中间的过程稍微繁琐,并没有顺序表方便
  • 所以今天我们也是用顺序表来实现一个栈
    在这里插入图片描述
    在这里插入图片描述

栈的操作

定义

说明:
top:是指向栈顶的位置,还是栈顶的下一个位置,在初始化讨论
capacity:表示栈的容量,因为是用顺序表实现,入栈的时候也需要判断是否现有元素个数是否达到容量
arr:就是个指向数组的指针

typedef int STDataType;
typedef struct ST
{int top;int capacity;STDataType* arr;
}ST;

初始化

说明:
初始化的时候我并没有给数组开辟空间,那么后面入栈的时候再进行扩容
因此capacity也就是0;
top:

  1. 这里我将top初始化为0,top指向的是栈顶的下一个元素
    • 此时top指向的是下标为4的位置
    • 而栈内的元素也是4,因此,我们可以用top直接表示栈内的元素个数(优势)
    • 当栈为空时,top为0
      在这里插入图片描述
  2. top指向栈顶的位置,当栈为空的时候top为-1;
    这种方式的优势在于可以将top直接当做下标
void STInit(ST* st)
{assert(st != NULL);st->arr = NULL;st->capacity = 0;st->top = 0;//指向栈顶的下一个元素
}

入栈

说明:
因为初始化的时候没有给数组开辟空间,所以入栈的时候要考虑是给栈开辟空间还是为栈扩容

void STPush(ST* st, STDataType x)
{assert(st != NULL);if (st->capacity == st->top){int capacity = st->capacity == 0 ? 4 : st->capacity * 2;STDataType* newp = (STDataType*)realloc(st->arr, sizeof(STDataType) * capacity);if (newp == NULL){perror("realloc fail");return;}st->arr = newp;st->capacity = capacity;newp = NULL;}st->arr[st->top++] = x;}

出栈

void STPop(ST* st)
{assert(st);if (!STEmpty(st)){st->top--;}
}

返回栈顶元素

因为这里的top表示栈顶的下一个位置,所以栈顶元素的下标是top-1

STDataType STTop(ST* st)
{assert(st);if (!STEmpty(st)){return st->arr[st->top - 1];}return -1;
}

判断栈是否为空

top为0就表示栈为空

bool STEmpty(ST* st)
{assert(st);return st->top == 0;
}

返回栈内元素个数

前面说了,top表示为栈顶元素的优点在于,top可以直接表示为元素个数

int STSize(ST* st)
{assert(st);return st->top;
}

销毁栈

void STDestory(ST* st)
{assert(st);free(st->arr);st->top = 0;st->capacity = 0;}

队列

队列的定义

从队的一头插入元素,另一头弹出元素
在这里插入图片描述

用顺序表实现还是用链表

  • 队列的特点是不符合顺序表来实现的,入队相当于顺序表的尾插,出队时,顺序表将下标为0的元素移除,那么只能将后面的数据依次向前移一位,出一次队就要将整个顺序表移动一次,这样的代价是非常大的。
  • 链表就很符合这个特点,入队就相当于尾插,出队就是头插,相比之下很方便,还是链表实现比较好

队列的操作

定义

用链表实现队列,那么就需要用到链表的这个结构体
队列想要入栈,是尾插,那么总不能每次插入一次就将链表遍历一遍寻找尾结点,所以,我们需要一个指针tail去记录尾结点
队列想要出栈,是头插,当然这只是原因之一,因为有了指向头结点的指针才能找到这个链表

typedef int QDataType;
typedef struct ListNode
{QDataType data;struct ListNode* next;
}ListNode;typedef struct Queue
{ListNode* head;ListNode* tail;
}Queue;

初始化

因为队列里还没有元素,指针都置空

void QueueInit(Queue* qu)
{assert(qu);qu->head = NULL;qu->tail = NULL;
}

入队

每一次入队都是先创建一个结点,在进行连接
这里要考虑队列为空的情况与不为空的情况是否可以共用一套代码

void QueuePush(Queue* qu, QDataType x)
{assert(qu);ListNode* new_node = (ListNode*)malloc(sizeof(ListNode));if (new_node == NULL){perror("malloc fail");return;}new_node->next = NULL;new_node->data = x;if (QueueEmpty(qu)){qu->head = new_node;qu->tail = new_node;}else{qu->tail->next = new_node;qu->tail = qu->tail->next;}
}

出队

先考虑队列为空的情况,
接着是不为空,不为空又分为一个结点和多个节点
分别去处理三种情况,尤其是要考虑只有一个结点的情况

void QueuePop(Queue* qu)
{assert(qu);if (!QueueEmpty(qu)){ListNode* tmp = qu->head;if (qu->head == qu->tail)//这个条件说明只有一个结点{free(tmp);tmp = NULL;qu->head = NULL;qu->tail = NULL;}else{qu->head = qu->head->next;free(tmp);tmp = NULL;}}
}

返回队头元素

不为空就返回head指向结点的数据,为空返回-1

QDataType QueueTop(Queue* qu)
{assert(qu);if (!QueueEmpty(qu)){return qu->head->data;}return -1;
}

返回队尾元素

QDataType QueueBack(Queue* qu)
{assert(qu);if (!QueueEmpty(qu)){return qu->tail->data;}return -1;
}

判断队列是否为空

判断是否为空很简单,只用判断两个指针是否同时为空
但也要考虑到意外状况,如果在函数外传参传错,导致两个指针其中一个为空,一个不为空,这种情况检查起来非常麻烦,所以在这里断言一下

bool QueueEmpty(Queue* qu)
{assert(qu);assert(!(qu->head == NULL && qu->tail != NULL));assert(!(qu->head != NULL && qu->tail == NULL));return qu->head == NULL && qu->tail == NULL;
}

返回队列元素个数

int QueueSize(Queue* qu)
{assert(qu);int size = 0;ListNode* cur = qu->head;while (cur != NULL){size++;cur = cur->next;}return size;
}

销毁队列

相当于销毁一个链表,最后函数结束时,手动将实参qu置空

void QueueDestory(Queue* qu)
{assert(qu);ListNode* cur = qu->head;while (cur != NULL){ListNode* nxt = cur->next;free(cur);cur = nxt;}qu->head = NULL;qu->tail = NULL;
}

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

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

相关文章

什么是无头浏览器以及其工作原理?

如果您对这个概念还不熟悉,那么使用无头网络浏览器的想法可能会让您感到不知所措。无头浏览器本质上与您熟悉的网络浏览器相同,但有一个关键区别:它们没有图形用户界面 (GUI)。这意味着没有按钮、选项卡、地址栏或视觉显示。 相反&#xff0c…

硬盘几个关键指标你一定要知道!

硬盘是数据中心中存储数据的重要部件,其关键指标影响硬盘的性能、可靠性和适用性。以下是一些常见的硬盘关键指标,并附上详细解释和举例: 容量(Capacity) 解释:硬盘的容量指其能存储数据的总量,…

CPN Tools学习——时间和队列【重要】

-Timed Color Sets 时间颜色集 -Token Stamps 令牌时间戳 -Event Clock 全局/事件/模拟时钟 -Time Delays on Transitions过渡的时间延迟 - List Color Set列表颜色集 - Queue排队 1.时间颜色集 在定时CPN模型令牌中有: (1)象征性的颜…

银河麒麟系统项目部署

使用服务器信息 软件:VMware Workstation Pro 虚拟机:ubtun 内存:20G 虚拟机连接工具: MobaXterm Redis连接工具: RedisDesktopManager 镜像:F:\Kylin-Server-10-8.2-Release-Build09-20211104-X86_64…

js: 百度云BOS 分片上传

百度云BOS存储后怎么查看或下载呢? // 1) 查看登录到百度智能云控制台 – 对象存储BOS”服务–选择一个Bucket,进入后可以查看该Bucket下的所有文件和文件夹。 2)下载OS浏览器端不支持批量下载,可以通过以下方式下载文件(使用BOS桌…

WindTerm使用SSH密钥连接阿里云实例,服务器设置SSH密钥登录

安装Windterm 地址https://github.com/kingToolbox/WindTerm/releases 下载完放到文件夹就可以打开 阿里云开启密钥对 打开阿里云ecs控制台 https://ecs.console.aliyun.com/keyPair/region/cn-wulanchabu 网络与安全->密钥对,创建密钥对,创建成…

记一次 .NET某工控视觉自动化系统 卡死分析

一:背景 1. 讲故事 今天分享的dump是训练营里一位学员的,从一个啥也不会到现在分析的有模有样,真的是看他成长起来的,调试技术学会了就是真真实实自己的,话不多说,上windbg说话。 二:WinDbg …

hbase安装部署

1,下载依赖包 最近的版本下载镜像 https://mirrors.aliyun.com/apache/hbase/ 比较旧的版本下载 https://archive.apache.org/dist/hbase 2,解压压缩包 3,更改配置 3-1 修改hbase-env.sh 必须加 JAVA_HOME export JAVA_HOME/usr/jdk64/jdk…

前端技术回顾系列 11|TS 中一些实用概念

在微信中阅读,关注公众号:CodeFit。 创作不易,如果你觉得这篇文章对您有帮助,请不要忘了 点赞、分享 和 关注 我的公众号:CodeFit,为我的持续创作提供动力。 上文回顾:泛型在类和接口中的应用 上一篇文章我们回顾了 泛型 在 类 和 接口 中的应用。 通过使用泛型,我们…

动态规划-简单多状态dp问题 -- 按摩师

动态规划-简单多状态dp问题 – 按摩师 文章目录 动态规划-简单多状态dp问题 -- 按摩师题目重现算法流程示例代码 题目重现 题目链接:按摩师 - 力扣 一个有名的按摩师会收到源源不断的预约请求,每个预约都可以选择接或不接。在每次预约服务之间要有休息时…

IINA for Mac v1.3.5 音视频软件 安装教程(保姆级)

Mac分享吧 文章目录 效果一、准备工作二、开始安装1、双击运行软件,将其从左侧拖入右侧文件夹中,等待安装完毕2、应用程序显示软件图标,表示安装成功 三、运行测试1、打开软件,测试2、查看版本号 **安装完成!&#xf…

C#——结构体详情

结构体 结构体也被称为结构类型(“structure type”或“struct type”),它是一种可封装数据和相关功能的值类型,在语法上结构体与类(class)非常相似,它们都可以用来封装数据,并且都…

【C语言】递归复杂度与链表OJ之双指针

【C语言】递归复杂度与链表OJ之双指针 🔥个人主页:大白的编程日记 🔥专栏:数据结构 文章目录 【C语言】递归复杂度与链表OJ之双指针前言一.递归复杂度1.1递归时间复杂度1.2递归空间复杂度 二.链表OJ之双指针2.1倒数第K个节点2.2链…

FastJson与JackSon 的碰撞。服务发送方与服务接收方字段不一致问题

情景再现 对接文档人家要求字段为 于是乎 咱就在服务的发送方定义参数字段为 服务接收方接收的类型为 later。。。。。 服务接收方接收到的参数字段 就不变成了fOrgId。跟外部系统对接就提示参数错误 原因 :lombok 在做set get的时候自动 无论你字段的首字母是大…

我与Python的一夜情

期末突击看这篇才够味! 环境搭建 首先就是相关工具的安装,直接搜就好,但是还是贴个网址吧: Welcome to Python.orghttps://www.python.org/ 然后就是根据自己的系统选择咯: 谁能闲来无事送我个mac玩玩 Windows的一…

哈喽GPT-4o——对GPT-4o 写论文的思考与看法

目录 几小时即可完成一份1万字论文的编写1、改写降重2、同义词替换降重3、避免连续相同4、缩写扩写降重5、关键词汇替换降重6、句式变换降重7、逻辑重组8、综合改写9、数据呈现方式变更10、概念解释降重 大家好,我是哪吒。 在ChatGPT4o对话框中输入:写一…

遥控玩具车电机驱动应用中的双H桥驱动芯片

遥控玩具车的基本工作原理是通过无线电遥控器发送信号,这些信号被玩具车内的接收器接收并解码,从而控制玩具车的运行。根据车身外型的不同,可以分为:普通的私家房车、越野车、货柜车、翻斗车等等。遥控器的操作,如前进…

测试 halcon measure_projection 算子

期望结果完全相同,但是下面的测试结果和halcon的差值如下: [0.132838, 0.231991, 0.265157, 0.296903, 0.0998573, 0.165907, 0.230686, 0.130266, 0.0977104, 0.197109, 0.198173, 0.197086, 0.190943, 0.177665, 0.163521, 0.146541, 0.161362, 0.166666, 0.2281…

NG32031单片机串口初始化

目录 1. 串口基础 2. 串口配置步骤 3. N32G031串口初始化示例 3.1开启时钟 3.2 配置GPIO 3.3 配置USART 3.4 使能中断(如果需要) 3.5. 示例代码 4. 调试和验证 5. 注意事项 6. 额外功能 NG32G031单片机的串口(UART)通常…

理解Python的元类

1.type()函数 type 函数是一个内置函数,用来获取一个对象的类型。它可以接受一个参数,返回这个参数的数据类型。type也可以用来创建类,type就是元类 x333 list["ab"] tuple (1, "a", True, 3.14) dict {name: Alice,…