当前位置: 首页 > news >正文

【数据结构与算法】队列和栈的相互实现以及循环队列

在这里插入图片描述

目录

  • 🌔一.用队列实现栈
    • 🌙1.题目描述
    • 🌙2.思路分析
    • 🌙3.代码实现
    • ⛈二.用栈实现队列
      • ☔1.题目描述
      • ☔2.思路分析
      • ☔3.代码实现
      • 🌈三.实现循环队列

🌔一.用队列实现栈

🌙1.题目描述

我们先看一下题目链接
在这里插入图片描述

🌙2.思路分析

我们知道栈是后进先出的线性表,题目要求我们使用两个队列来实现,队列的特性是先进先出,从队尾进 队头出,图解如下:
在这里插入图片描述
那我们如何向队列进元素1 2 3 4出队列是4 3 2 1的顺序呢。假设现在队列中有1 2 3 4:
在这里插入图片描述
我们想让4先出队列的话,可以将前面的1 2 3先转移到另一个队列中,再出4,以此类推实现后进先拿出。使得一个队列为空用来出队列时倒元素的目的地,另一个不为空的队列用来插入数据。想要返回栈顶元素 只要返回非空队列队尾元素即可。

🌙3.代码实现

代码的实现,我们需要先手撕个队列包括队列的基本功能并且定义两个队列:

typedef int QDataType;typedef struct QueueNode
{struct QueueNode* next;QDataType data;
}QNode;typedef struct Queue
{QNode* head;QNode* tail;int size;
}Queue;typedef struct {Queue q1;Queue q2;
} MyStack;
void QueueInit(Queue* pt);
void QueueDestroy(Queue* pt);void QueuePush(Queue* pt, QDataType x);
void QueuePop(Queue* pt);int QueueSize(Queue* pt);bool QueueEmpty(Queue* pt);QDataType QueueFront(Queue* pt);
QDataType QueueBack(Queue* pt);void QueueInit(Queue* pt)
{assert(pt);pt->head = pt->tail = NULL;pt->size = 0;
}void QueueDestroy(Queue* pt)
{assert(pt);QNode* cur = pt->head;while (cur){QNode* next = cur->next;free(cur);cur = next;}pt->head = pt->tail = NULL;pt->size = 0;
}void QueuePush(Queue* pt, QDataType x)
{QNode* newnode = (QNode*)malloc(sizeof(QNode));if (newnode == NULL){perror("malloc fail");return;}newnode->next = NULL;newnode->data = x;if (pt->head == NULL){assert(pt->tail == NULL);pt->head = pt->tail = newnode;}else{pt->tail->next = newnode;pt->tail = newnode;}pt->size++;}void QueuePop(Queue* pt)
{assert(pt);assert(pt->head != NULL);if (pt->head->next == NULL){free(pt->head);pt->head = pt->tail = NULL;}else{QNode* next = pt->head->next;free(pt->head);pt->head = next;}pt->size--;
}int QueueSize(Queue* pt)
{assert(pt);return pt->size;
}bool QueueEmpty(Queue* pt)
{assert(pt);return pt->size == 0;
}QDataType QueueFront(Queue* pt)
{assert(pt);assert(!QueueEmpty(pt));return pt->head->data;
}
QDataType QueueBack(Queue* pt)
{assert(pt);assert(!QueueEmpty(pt));return pt->tail->data;
}

紧接着即使栈的实现:

MyStack* myStackCreate() {MyStack*ptr=(MyStack*)malloc(sizeof(MyStack));if(ptr==NULL){perror("malloc fail");return NULL;}QueueInit(&ptr->q1);QueueInit(&ptr->q2);return ptr;}void myStackPush(MyStack* obj, int x) {if(!QueueEmpty(&obj->q1))//向非空队列插入数据{QueuePush(&obj->q1,x);}else{QueuePush(&obj->q2,x);}}int myStackPop(MyStack* obj) {Queue* empty=&obj->q1;//先假设空与非空队列Queue* noempty=&obj->q2;if(!QueueEmpty(&obj->q1)){empty=&obj->q2;noempty=&obj->q1;}while(QueueSize(noempty)>1){QueuePush(empty,QueueFront(noempty));QueuePop(noempty);}int ret=QueueFront(noempty);QueuePop(noempty);return ret;}int myStackTop(MyStack* obj) {if(!QueueEmpty(&obj->q1)){return QueueBack(&obj->q1);}else{return QueueBack(&obj->q2);}
}bool myStackEmpty(MyStack* obj) {return QueueEmpty(&obj->q1)&&QueueEmpty(&obj->q2);//只有当两队列都为空的时候栈才为空
}void myStackFree(MyStack* obj) {QueueDestroy(&obj->q1);QueueDestroy(&obj->q2);free(obj);
}

⛈二.用栈实现队列

☔1.题目描述

我们先看一下题目链接
在这里插入图片描述

☔2.思路分析

这题相反,需要我们实现两个栈来实现队列。究其本质就是使用两个后进先出显示先进先出,我们用图例来理解下:
在这里插入图片描述
入栈1 2 3 4后出栈顺序为4 3 2 1我们可以依次顺序入到另一个栈中,再将数据从栈中弹出,即可做到出栈1 2 3 4的顺序。图示如下:
在这里插入图片描述

☔3.代码实现

代码还是需要先手撕个栈以及实现栈的基本功能并定义两个栈分别用来插入和删除:

typedef int STDataType;typedef struct stack
{STDataType* a;STDataType top;STDataType capacity;
}ST;void InitST(ST* s);//初始化栈
void DestroyST(ST* s);//销毁栈void Push(ST* s, int x);//压栈
void Pop(ST* s);//出栈bool STEmpty(ST* s);//判断栈是否为空栈
STDataType STSize(ST* s);//当前栈的元素个数
STDataType STTop(ST* s);//返回栈顶元素
void InitST(ST* s)
{assert(s);s->a = (STDataType*)malloc(sizeof(STDataType) * 4);if (s->a == NULL){perror("malloc fail");return;}s->capacity = 4;s->top = -1;///top记录指向的当前元素
}void DestroyST(ST* s)
{assert(s);s->capacity = 0;free(s->a);s->a = NULL;s->top = -1;
}void Push(ST* s,int x)
{assert(s);if (s->top+1 == s->capacity){s->a = (STDataType*)realloc(s->a,sizeof(STDataType)*s->capacity*2);if (s->a == NULL){perror("malloc fail");return;}s->capacity *= 2;}s->a[s->top+1] = x;s->top++;}void Pop(ST* s)
{assert(s);assert(!STEmpty(s));s->top--;
}bool STEmpty(ST* s)
{assert(s);if (s->top == -1)return true;return false;
}STDataType STSize(ST* s)
{assert(s);return s->top + 1;
}STDataType STTop(ST* s)
{assert(s);return s->a[s->top];
}typedef struct {ST popst;ST pushst;
} MyQueue;

定义时我们就指定了用来插入的栈和删除的栈,我们在创建好自己的队列后,对队列进行插入操作时,只需要向pushst栈插入数据即可,另一个popst栈保持为空,在需要出队列时,将另一个栈的数据倒过来,再出数据 即可保证先进先出的顺序。

MyQueue* myQueueCreate() {MyQueue* ptr=(MyQueue*)malloc(sizeof(MyQueue));if(ptr == NULL){perror("malloc fail");return NULL;}InitST(&ptr->popst);InitST(&ptr->pushst);return ptr;}void myQueuePush(MyQueue* obj, int x) {Push(&obj->pushst,x);
}int myQueuePop(MyQueue* obj) {int ret=myQueuePeek(obj);Pop(&obj->popst);return ret;
}int myQueuePeek(MyQueue* obj) {if(STEmpty(&obj->popst)){while(!STEmpty(&obj->pushst)){Push(&obj->popst,STTop(&obj->pushst));Pop(&obj->pushst);           }}return STTop(&obj->popst);
}bool myQueueEmpty(MyQueue* obj) {return STEmpty(&obj->popst)&&STEmpty(&obj->pushst);
}void myQueueFree(MyQueue* obj) {DestroyST(&obj->popst);DestroyST(&obj->pushst);free(obj);}

🌈三.实现循环队列

线性表有顺序存储和链式存储,栈是线性表,具有这两种存储方式。同样,队列作为一种特殊的线性表,也同样存在两种存储方式。队列的链式存储即是进行尾插头删的单链表,所以我们讲解队列的顺序存储———循环队列。
借助题目更好的讲解循环队列题目链接
在这里插入图片描述
首先我们创建结构体用来存储数据,其中包含指向存储数据地址的指针记录队列有效长度的整形变量,还有两个记录循环队列头和尾数组下标的变量。

typedef struct {int *a;int rear;int front;int k;} MyCircularQueue;

接着我们初始化队列,对于给定的有效长度k我们malloc比k多一的存储空间,方便表示存储满的情况另一种解决方法是增加size变量 用于记录当前数据个数。

MyCircularQueue* myCircularQueueCreate(int k) {MyCircularQueue* obj=(MyCircularQueue*)malloc(sizeof(MyCircularQueue));obj->rear=obj->front=0;obj->k=k;obj->a=(int*)malloc(sizeof(int)*(k+1));return obj;
}

队列为空的即是头尾指针指向同一地方的情况。

bool myCircularQueueIsEmpty(MyCircularQueue* obj) {return obj->front == obj->rear;
}

判断队列是否满,由于rear有可能比front大,也可能小,所以尽管他们只相差一个位置时就是满的情况,但可能相差整整一圈。为了避免判断错误,则需要取模判断。

bool myCircularQueueIsFull(MyCircularQueue* obj) {return (obj->rear+1)%(obj->k+1) == obj->front;
}

剩余代码

bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {if(myCircularQueueIsFull(obj))return false;obj->a[obj->rear++]=value;obj->rear%=(obj->k+1);return true;
}bool myCircularQueueDeQueue(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj))return false;obj->front++;obj->front%=(obj->k+1);return true;
}int myCircularQueueFront(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj))return -1;return obj->a[obj->front];
}int myCircularQueueRear(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj))return -1;int x=obj->rear == 0? obj->k:obj->rear-1;//防止数据存储到最后一位return obj->a[x];
}void myCircularQueueFree(MyCircularQueue* obj) {free(obj->a);free(obj);
}

相关文章:

【数据结构与算法】队列和栈的相互实现以及循环队列

目录🌔一.用队列实现栈🌙1.题目描述🌙2.思路分析🌙3.代码实现⛈二.用栈实现队列☔1.题目描述☔2.思路分析☔3.代码实现🌈三.实现循环队列🌔一.用队列实现栈 🌙1.题目描述 我们先看一下题目链接…...

mysql连接不上问题解决

公司新搭内网测试环境,mysql远程登录问题解决 远程登录: 1 修改host, mysql> select user,host,plugin from user; ---------------------------------------------------- | user | host | plugin | ------------------------…...

利用nginx实现动静分离的负载均衡集群实战

前言 大家好,我是沐风晓月,今天我们利用nginx来作为负载,实现两台apache服务器的动静分离集群实战; 本文收录于沐风晓月的专栏《linux基本功-系统服务实战》,更多内容可以关注我的博客: https://blog.csd…...

与chatGPT神聊,引领你深入浅出系统调用

在操作系统的教学中,系统调用的作用不言而喻,但是,对系统调用常常是雾里看花,似乎明白,又难以真正的触及,即使在代码中调用了系统调用,比如调用fork()创建进程&#xff0…...

自学大数据第十天~Hbase

随着数据量的增多,数据的类型也不像原来那样都是结构化数据,还有非结构化数据; Hbase时google 的bigtable的开源实现, BigtableHbase文件存储系统GFSHDFS海量数据处理MRMR协同管理服务chubbyzookeeper虽然有了HDFS和MR,但是对于数据的实时处理是比较困难的,没有办法应对数据的…...

vue更高效的工具-vite

目录 1.webpack 2.vite是什么 3.使用vite创建项目 4.最后总结 🐼webpack 简单来说,Webpack是一个打包工具。 站在2018年的角度,成为一个优秀的前端工程师,除了要会写页面样式和动态效果之外,还需要会用主流的单页…...

HFish蜜罐的介绍和简单测试(一)

目录 0、什么是蜜罐 0.1、蜜罐的定义 0.2、蜜罐的优势 0.3、蜜罐与情报 1、HFish介绍 1.1、设计理念 1.2、HFish架构 1.3、HFish特点 1.4、常见蜜罐场景 2、快速部署 2.1、环境要求 2.2、联网环境,一键安装 2.3、安装效果 3、错误排查 3.1、管理端问题…...

2023面试题汇总二

一、CSS面试题 1. 清除浮动的方式有哪些? 为什么要清除浮动?因为浮动的盒子脱离标准流,如果父盒子没有设置高度的话,下面的盒子就会撑上来。 额外标签法(在最后一个浮动标签后,新加一个标签,给其设置cle…...

C# 支付宝接口在线收款退款

收款 在C#中使用支付宝在线支付功能,需要使用支付宝开放平台提供的SDK(软件开发工具包),通过SDK中提供的API(应用程序接口)实现在线支付功能。 以下是使用C#实现支付宝在线支付的大致步骤: 获…...

python例程:《企业编码生成系统》程序

目录《企业编码生成系统》程序使用说明主要代码演示源码及说明文档下载路径《企业编码生成系统》程序使用说明 在PyCharm中运行《企业编码生成系统》即可进入如图1所示的系统主界面。在该界面中可以选择要使用功能对应的菜单进行不同的操作。在选择功能菜单时,只需…...

基于EB工具的TC3xx_MCAL配置开发04_ADC模块软件触发Demo配置

目录 1.概述2. EB配置2.1 添加HwUnit2.2 AdcPrescale配置2.3 添加ADC通道2.4 添加Adc Group2.5 Adc Group配置2.5.1 AdcGroup->General2.5.2 AdcGroup->AdcNotification2.5.3 AdcGroup->AdcGroupDefinition2.5.4 AdcGroup->AdcResRegDefinition2.6 中断配置1.概述 …...

内存操作函数

前言 🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨ 🐻推荐专栏: 🍔🍟🌯 c语言进阶 🔑个人信条: 🌵知行合一 🍉本篇简介:>:介绍c语言中有关指针更深层的知识. 金句分享: ✨未来…...

免费搭建个人博客

免费搭建个人博客,并发布到公网 利用hexo搭建个人博客,通过gitee的pages发布到公网 1 前置准备 安装git、安装node.js(尽量选择长期支持的版本) node.js官网:https://nodejs.org/en/ git官网:https://git-scm.com/book/zh/v2 安装…...

【Vue全家桶】详解Vue Router(一)

【Vue全家桶】Vue Router详解(一) Vue系列文章目录: 内容参考链接Vue(一)【Vue全家桶】邂逅Vue、Vue的多种引入方式Vue(二)【Vue全家桶】声明式编程、MVVMVue(三)【Vue…...

9大 HIVE SQL 最频繁被问到的面试题

SQL是用于数据分析和数据处理的最重要的编程语言之一,因此与数据科学相关的工作(例如数据分析师、数据科学家和数据工程师)在面试时总会问到关于 SQL 的问题。 SQL面试问题旨在评估应聘者的技术和解决问题的能力。因此对于应聘者来说&#x…...

学大数据算跟风吗?

随着互联网、物联网和人工智能等技术的不断发展,大数据技术逐渐进入人们的视野,成为一个备受关注的热点话题。那么,大数据专业好学吗?前景如何?下面我们来一起探讨一下。 一、大数据专业的学习难度 大数据技术是一种综…...

[C#]关于ListBox的坑

在用C#写个Demo程序的时候,使用ListBox保存读取到的每行内容。然后在重复读取的时候需要清除ListBox中的内容,我看到ListBox有清除的函数:lsbFiles.Items.Clear();以为这个函数就可以解决所有的问题,但是....于是想了个方法准备一…...

Apache POI 入门·第一话

文章目录1 摘要2 Apache POI2.1 介绍2.2 应用场景2.3 入门案例2.3.1 将数据写入Excel文件2.3.1.1 导入POI maven坐标2.3.1.2 代码开发2.3.1.3 实现效果2.3.2 读取Excel文件中的数据2.3.3 实现效果2.4 开发案例——导出运营数据Excel报表2.4.1 产品原型2.4.2 接口设计2.4.3 代码…...

8个python自动化脚本提高打工人幸福感~比心~

人生苦短,我用Python 最近有许多打工人都找我说打工好难 每天都是执行许多重复的任务, 例如阅读新闻、发邮件、查看天气、打开书签、清理文件夹等等, 使用自动化脚本,就无需手动一次又一次地完成这些任务, 非常方便…...

【嵌入式烧录/刷写文件】-1-详解Motorola S-record(S19/SREC/mot/SX)格式文件

目录 1 什么是Motorola S-record 2 Motorola S-record的格式 2.1 Motorola S-record的结构 2.1.1 “Record type记录类型”的说明 2.1.2 “Record length记录长度”的说明 2.1.3 如何计算“Checksum校验和” 2.2 Record order记录顺序 2.3 Text line terminator文本行终…...

图形视图界面 图形效果

Qt的标准图形效果类: QGraphicsBlurEffect提供模糊效果QGraphicsColorizeEffect提供染色效果QGraphicsDropShadowEffect提供阴影效果QGraphicsOpacityEffect提供透明效果 QGraphicsBlurEffect(模糊效果) 模糊效果会模糊源。此效果对于减少细…...

ElementUI学习笔记

目录 一、简单介绍 二、安装 1、下载 2、引入 三、布局 1、简介 2、使用 3、好处 四、布局容器 1、常见排布 2、调整样式 五、按钮 1、简单引用 2、改变样式 3、加载中效果 六、表格 1、简单使用 2、样式修改 七、对话框 1、简单使用 2、添加自定义内容 3、…...

安装KVM并创建虚拟机及基本使用

#环境说明:Centos7 环境准备: CPU开启虚拟化,给宿主机的CPU和内存分配足够多的配置 安装KVM 1.安装相关软件包 yum -y install qemu-kvm libvirt virt-manager virt-install virt-viewer 软件包简介: qemu-kvm: 为kvm提供…...

一种LCD屏闪问题的调试

背景 项目使用ESP32-S3 RGB接口驱动的LCD, 框架 idf-v5.0, LVGL-v7.11 显示画面正常, 但肉眼可见的像是背光在闪烁, 背光电路是应用很久的经典电路, 且排查背光驱动无错, 但开机一段时间后, 闪烁会明显减轻 记录 这块屏的显示驱动芯片为ST7701S, 查看芯片手册有说明特定的上…...

Java程序运行在Docker等容器环境有哪些新问题?

第30讲 | Java程序运行在Docker等容器环境有哪些新问题? 如今,Docker 等容器早已不是新生事物,正在逐步成为日常开发、部署环境的一部分。Java 能否无缝地运行在容器环境,是否符合微服务、Serverless 等新的软件架构和场景&#x…...

C语言面试最常问的三个关键字

文章目录前言一,static关键字的作用二,const 关键字的作用2.1, 修饰局部变量2.2,修饰指针2.3, 修饰函数形参2.4,修饰函数的返回值三,volatile关键字的作用前言 面试的时候,C语言最常…...

【Linux】-初识Linux

作者:学Java的冬瓜 博客主页:☀冬瓜的主页🌙 专栏:【Linux】 分享:逆着光行走,任风吹雨打。 ——《起风了》 主要内容:Linux的一些最基本指令,Linux的小程序,Linux关于连…...

精选7个 Python 学习资源库,助你成为优秀的开发者

当你在学习编程时,很容易被大量的资源所吓到,不知道该从何开始。 GitHub 仓库是一个很好的起点,因为它们提供了一种非常实用的方式来了解实际的编程应用。你可以查看其他人的代码,并将其与自己的代码进行比较和学习。 当涉及到 …...

【大数据处理与可视化】三 、Pandas库的运用

【大数据处理与可视化】三 、Pandas库的运用实验目的实验内容实验步骤一、使用pandas库分别创建Series对象和DataFrame对象,并对创建的对象使用索引、排序等相关操作;练习DataFrame对象的统计计算和统计描述的功能。1&2、创建一个DataFrame(d…...

FPGA解码SDI视频任意尺寸缩放拼接输出 提供工程源码和技术支持

目录1、前言2、SDI理论练习3、设计思路和架构SDI摄像头Gv8601a单端转差GTX解串SDI解码VGA时序恢复YUV转RGB图像缩放FDMA图像缓存实现拼接HDMI驱动4、vivado工程详解5、上板调试验证并演示6、福利:工程代码的获取1、前言 FPGA实现SDI视频编解码目前有两种方案&#…...

东莞网络推广服务平台/安徽网站优化

官方网站:http://www.xfyun.cn/ 注册还要绑定微信,坑啊,识别率感觉没得微信语音好,但是微信语音审核一直不过,研究下这个 1.下载sdk,主要就下面几个文件,我主要用的是语音识别 2.导入相关的库 3…...

wordpress wp config/百度平台我的订单

参考资料:OpenDDS环境搭建-Windows 10_Saint_EK的博客-CSDN博客_opendds环境搭建由于项目中需要使用到OpenDDS,故记录下环境搭建过程。安装环境windows 10 64位已安装 Visual Stdio 2017、Visual Stdio 2019安装步骤总述:需安装Perl、ACETAO、…...

安阳区号码/青岛seo整站优化哪家专业

第一步 收集并下载xp的安全补丁 在微软里一个一个下载?不但麻烦,而且还会遗漏,我们用个能批量下载所有补丁的软件Windows Updates Downloader 点击下载。该软件需要Microsoft .NET Framework 2.0支持点击下载。安装Windows Updates Downloade…...

购物网站建设与开发/近期国内外重大新闻10条

好吧,我标题党了。作为 Python 教师,我发现理解装饰器是学生们从接触后就一直纠结的问题。那是因为装饰器确实难以理解!想弄明白装饰器,需要理解一些函数式编程概念,并且要对Python中函数定义和函数调用语法中的特性有…...

如果做游戏的技术用来做网站/国内推广平台有哪些

这里写的是关于FOXMAIL点“发送”按钮无反应的问题,这是因为FOXMAIL邮箱文件夹有2G的容量限制导致。解决办法如下: 一个文件箱中的单个文件达到2G的时候也就爆了,网络上有很多的教程来处理2G容量的限制,其实很简单的,没…...

济南城市建设集团/深圳整站seo

我有一个本地安装的库的问题 . 在我的项目中有xmlrpc 0.7库:myproject/-- xmlrpc0.7/-- src/如果没有找到,我希望CMake使用本地xmlrpc 0.7目录进行回退 . 两个问题,第一个,find_path()或find_library()不适用于本地目录 . 我使用了…...