栈和队列特别篇:栈和队列的经典算法问题

图均为手绘,代码基于vs2022实现

系列文章目录

数据结构初探: 顺序表
数据结构初探:链表之单链表篇
数据结构初探:链表之双向链表篇
链表特别篇:链表经典算法问题
数据结构:栈篇
数据结构:队列篇


文章目录

  • 系列文章目录
  • 前言
  • 一.有效的括号(leetcode 20)
  • 二.用队列实现栈(leetcode 225)
  • 三.用栈实现队列(leetcode 232)
  • 四.设计循环队列(leetcode 622)
  • 五.总结


前言

  • 栈和队列作为基础数据结构,是算法设计中的重要基石。它们在操作系统、编译器设计、网络协议等领域有着广泛应用。本文将通过C语言实现经典算法问题,帮助读者深入理解它们的底层原理和应用技巧。学习这些内容不仅能提升算法能力,还能加深对内存管理和数据结构设计的理解。

一.有效的括号(leetcode 20)

让我们先来看看题目:
在这里插入图片描述
让我们来理清楚思路:

  • 首先,我们可以用栈来实现匹配

1.我们将左括号入栈;
2.在遍历中找到右括号,进行匹配;

  • 在这个过程中,我们需要手撕一个栈的代码出来,加在题目代码之前,如果发现还不太熟练,可以再去我的这篇博客中熟悉熟悉—> 数据结构:栈篇
  • 我们还是上代码实际感受一下吧(考虑到篇幅有限,已省略栈的代码):
//注意这里是字符,栈的代码中要将int改为char
//typedef char STDataType;
bool isValid(char* s) {ST st;//创建栈STInit(&st);//初始化栈while(*s)//依次遍历字符串{if(*s == '(' || *s == '[' || *s == '{')//如果是左括号{STPush(&st,*s);//我们就将其入栈}else//其他情况{if(STEmpty(&st))//如果没找到左括号,就算下面有右括号{//我们也无法找到匹配的括号STDestroy(&st);//所以销毁,防止内存泄漏return false;//直接返回false,表示找不到匹配的括号}//因为题目要求的是按顺序一一对应;char top=STTop(&st);//此时记录下栈顶元素STPop(&st);//再pop,表示出栈操作//匹配括号逻辑结构//大家可以自行分析;//即如果右括号没有在栈顶找到对应的左括号,则错误if((*s == ')' && top != '(')||(*s == ']' && top != '[')||(*s == '}' && top != '{')){STDestroy(&st);//所以销毁,防止内存泄漏return false;//直接返回false,表示找不到匹配的括号}}s++;//更新循环条件}bool ret=STEmpty(&st);//记录是否找到对应括号的状态STDestroy(&st);//销毁,防止内存泄漏return ret;//返回对应状态
}

在这里插入图片描述

学完后,自己多多练习,自行手撕,理清逻辑即可;

二.用队列实现栈(leetcode 225)

让我们先来看看题目:
在这里插入图片描述
让我们来理清楚思路:
我们需要用到两个队列来实现栈:

  • 一个用来当作入栈队列
  • 一个用来当作出栈队列
  • 我们还要注意其LIFO的性质,和随进随出的特点
    1.我们需要保持一个队列为空,一个队列存数据
    2.出栈,把前面的数据导入空队列
    3.如此两班来回倒,就可以实现栈的特性;
    逻辑如图:
    在这里插入图片描述
    在这个过程中,我们需要手撕队列的代码出来,加在题目代码之前,如果发现还不太熟练,可以再去我的这篇博客中熟悉熟悉—> 数据结构:队列篇
    我们来实战一下:
//创建两个队列的结构体
typedef struct {Queue q1;Queue q2;
} MyStack;
//我们需要动态开辟出空间来实现mystack
MyStack* myStackCreate() {MyStack* pst=(MyStack*)malloc(sizeof(MyStack));if(pst == NULL){perror("malloc fail");return NULL;}QueueInit(&pst->q1);//对两个队列初始化QueueInit(&pst->q2);return pst;//返回初始化后的我们的mystack
}
//来实现栈的插入
void myStackPush(MyStack* obj, int x) {if(!QueueEmpty(&obj->q1))//如果q1不为空就插入到q1里{//反正有一个队列里面是空的,我们不能插入,一旦插入就会弄乱整体的逻辑QueuePush(&obj->q1,x);//可以参看上文图进行理解}else//如果q2不为空就插入到q2里{//两个都为空就随便选一个QueuePush(&obj->q2,x);}
}
//实现栈的删除,题目要求:移除并返回栈顶元素
int myStackPop(MyStack* obj) {//首先要明确哪个为空,哪个为非空Queue* emptyQ=&obj->q1;//假设q1为空Queue* nonemptyQ=&obj->q2;//q2不为空if(!QueueEmpty(&obj->q1))//验证是否正确{//进入则是不正确emptyQ=&obj->q2;//不正确则交换nonemptyQ=&obj->q1;}while(QueueSize(nonemptyQ) > 1)//设置遍历循环{QueuePush(emptyQ,QueueFront(nonemptyQ));//将非空的前面n-1个挪入空队列QueuePop(nonemptyQ);//pop掉已经挪入的那n-1个}//剩下的就是要按照栈的特性LIFO顺序的数据int top=QueueFront(nonemptyQ);//按照题目要求记录下数据QueuePop(nonemptyQ);//popreturn top;//返回数据
}
//返回栈顶元素
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);
}

销毁逻辑示例图:
在这里插入图片描述

在这里插入图片描述

三.用栈实现队列(leetcode 232)

让我们先来看看题目:
在这里插入图片描述
让我们来理清楚思路:
我们需要用到两个栈来实现队列:

  • 一个用来当作入队栈
  • 一个用来当作出队栈
  • 我们还要注意其FIFO的性质,和随进随出的特点
    1.我们需要保持一个入队栈,一个出队栈
    2.出队列,把前面的数据导入空栈
    3.当出队栈不为空时,不能再将出队栈中的数据导入,否则顺序会乱;
    如图:
    在这里插入图片描述
    让我们来感受一下代码:
typedef struct {ST pushst;//入队栈ST popst;//出队栈
} MyQueue;int myQueuePeek(MyQueue* obj);//返回队头元素的函数声明;
//动态开辟
MyQueue* myQueueCreate() {MyQueue* obj=(MyQueue*)malloc(sizeof(MyQueue));if(obj==NULL){perror("malloc fail");return NULL;}STInit(&obj->pushst);//初始化两个栈STInit(&obj->popst);return obj;//返回模拟队列
}void myQueuePush(MyQueue* obj, int x) {STPush(&obj->pushst,x);//正常往入队栈中直接插入
}
//按题目要求,从队列的开头移除并返回元素
int myQueuePop(MyQueue* obj) {int front=myQueuePeek(obj);//记录队头元素STPop(&obj->popst);//删除return front; //返回元素
}int myQueuePeek(MyQueue* obj) {if(STEmpty(&obj->popst))//如果出队栈为空{while(!STEmpty(&obj->pushst))//开始遍历循环{//并且入队栈不为空STPush(&obj->popst,STTop(&obj->pushst));//将全部元素挪过去STPop(&obj->pushst);//在入队栈中删除的操作}}return STTop(&obj->popst);//返回栈顶即是队头,可以参看上图;
}
//判空
bool myQueueEmpty(MyQueue* obj) {
//两个栈都为空,才为空return STEmpty(&obj->pushst) && STEmpty(&obj->popst);
}
//释放所有动态开辟的空间
void myQueueFree(MyQueue* obj) {STDestroy(&obj->pushst);STDestroy(&obj->popst);free(obj);}

在这里插入图片描述

四.设计循环队列(leetcode 622)

让我们来看看题目:
在这里插入图片描述
我们来理清楚逻辑:
这里我们选择用数组来实现,链表实现也是可以的,但是相对来说,数组实现会更加容易;对于数组是否满了,我们有两种解决办法:

  • 在结构体中加入size来计量;
  • 空出一个位置来提醒判断已满
    让我们来实战:
//定义结构体
typedef struct {int* a;//静态数组int front;//头int rear;//尾int k;//满数据时候的个数
} MyCircularQueue;
bool myCircularQueueIsEmpty(MyCircularQueue* obj);//函数声明,防止编译错误
bool myCircularQueueIsFull(MyCircularQueue* obj);
//开出对应大小的空间
MyCircularQueue* myCircularQueueCreate(int k) {MyCircularQueue* obj=(MyCircularQueue*)malloc(sizeof(MyCircularQueue));if(obj==NULL){perror("malloc fail");return NULL;}obj->a=(int*)malloc(sizeof(int)*(k+1));//多开,防止溢出if(obj->a==NULL){perror("malloc fail");return NULL;}obj->front=obj->rear=0;//指向开头obj->k=k;return obj;
}
//插入
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {if(myCircularQueueIsFull(obj))//如果是满的{return false;//返回false,表示不能再插入了}obj->a[obj->rear++] = value;//否则存储值,并且rear++到下一个位置;obj->rear %= (obj->k+1);//对尾取模,比如,1%5==1,3%5==3,如果满了5%5==0//就会循环回到数组开始的下标return true;//返回true,表示还可以插入
}
//删除
bool myCircularQueueDeQueue(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj))//如果是空的{return false;//返回false,表示不能再删除了}obj->front++;//头往下一个位置走,这个就与我们曾经讲过的顺序表里面的删除一样的//只要我们不把它计量在有效个数中就可以删除它obj->front %= (obj->k+1);//对头取模,比如,1%5==1,3%5==3,如果满了5%5==0//就会循环回到数组开始的下标return true;//返回true,表示还可以删除
}
//按题目要求:从队首获取元素。如果队列为空,返回 -1
int myCircularQueueFront(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj))//若为空{return -1;//返回}return obj->a[obj->front];//返回头元素
}
//按题目要求:获取队尾元素。如果队列为空,返回 -1 
int myCircularQueueRear(MyCircularQueue* obj) {if(myCircularQueueIsEmpty(obj)){return -1;}return obj->a[(obj->rear-1+obj->k+1) % (obj->k+1)];//防止因为rear在下标0的位置--,而发生错误;//通过取模,来造成循环;
}
//简单的判空
bool myCircularQueueIsEmpty(MyCircularQueue* obj) {return obj->rear == obj->front;
}
//判断满
bool myCircularQueueIsFull(MyCircularQueue* obj) {//通过取模,来造成循环;return (obj->rear+1) % (obj->k+1) == obj->front;//保证rear的下一个是头
}
//释放,相同与前两题,可以自行画图理解;
void myCircularQueueFree(MyCircularQueue* obj) {free(obj->a);free(obj);
}

以上就是我要讲的所有的经典问题,你学会了吗?


五.总结

在本次博客中,我们围绕栈和队列这两种基础数据结构,通过C语言实现经典算法问题,深入探索了它们的底层原理与应用技巧。

  • 首先是“有效的括号”问题,利用栈来匹配括号,左括号入栈,右括号与栈顶元素匹配,这种方式充分展现了栈“后进先出”(LIFO)的特性,在处理具有成对结构的数据时十分有效 ,能够清晰地判断括号是否匹配,避免逻辑混乱。
  • 接着,在“用队列实现栈”和“用栈实现队列”的问题中,分别运用两个队列和两个栈来模拟对方的特性。用队列实现栈时,通过在两个队列间来回转移数据,确保满足栈的LIFO性质;用栈实现队列则是利用两个栈,在入队栈和出队栈间合理转移数据,保证队列“先进先出”(FIFO)的特性。这两个问题不仅锻炼了对数据结构特性的灵活运用,还让我们深入理解了不同数据结构间的转换和模拟方法。
  • 最后,在“设计循环队列”中,使用数组实现循环队列,通过巧妙的取模操作实现队列的循环,同时提供了判空和判满的方法。这种实现方式既高效又简洁,充分利用了数组的连续性和可索引性,加深了我们对队列数据结构的理解和应用能力。

通过解决这些经典算法问题,我们不仅掌握了栈和队列在实际编程中的应用,还提升了算法设计、逻辑思维以及内存管理的能力。希望读者能够通过不断练习,熟练掌握这些知识,在后续的编程学习和实践中更加得心应手,灵活运用栈和队列解决各种复杂的实际问题,为更深入的算法学习和系统开发打下坚实的基础。

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

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

相关文章

使用 OpenResty 构建高效的动态图片水印代理服务20250127

使用 OpenResty 构建高效的动态图片水印代理服务 在当今数字化的时代,图片在各种业务场景中广泛应用。为了保护版权、统一品牌形象,动态图片水印功能显得尤为重要。然而,直接在后端服务中集成水印功能,往往会带来代码复杂度增加、…

C++并行化编程

C并行化编程 C 简介 C 是一种静态类型的、编译式的、通用的、大小写敏感的、不规则的编程语言,支持过程化编程、面向对象编程和泛型编程。 C 被认为是一种中级语言,它综合了高级语言和低级语言的特点。 C 是由 Bjarne Stroustrup 于 1979 年在新泽西州美…

Java开发vscode环境搭建

1 几个名词 JDK Java Development Kit JRE Java Runtion Environment JVM JDK 包括 Compiler,debugger,JRE等。JRE包括JVM和Runtime Library。 2 配置环境 2.1 安装JDK 类比 C/C的 g工具 官网:https://www.oracle.com/java/technologies/downloads/ 根据自己使…

pytorch基于FastText实现词嵌入

FastText 是 Facebook AI Research 提出的 改进版 Word2Vec,可以: ✅ 利用 n-grams 处理未登录词 比 Word2Vec 更快、更准确 适用于中文等形态丰富的语言 完整的 PyTorch FastText 代码(基于中文语料),包含&#xff1…

riscv xv6学习笔记

文章目录 前言util实验sleeputil实验pingpongutil实验primesxv6初始化代码分析syscall实验tracesyscall实验sysinfoxv6内存学习笔记pgtbl实验Print a page tablepgtbl实验A kernel page table per processxv6 trap学习trap实验Backtracetrap实验Alarmlazy实验Lazy allocationxv…

FFmpeg(7.1版本)编译:Ubuntu18.04交叉编译到ARM

一、本地编译与交叉编译 1.本地编译 ① 本地编译:指的是在目标系统上进行编译的过程 , 生成的可执行文件和函数库只能在目标系统中使用。 如 : 在 Ubuntu中,本地编译的可执行文件只能在Ubuntu 系统中执行 , 无法在 Windows / Mac / Android / iOS 系…

创新创业计划书|建筑垃圾资源化回收

目录 第1部分 公司概况........................................................................ 1 第2部分 产品/服务...................................................................... 3 第3部分 研究与开发.................................................…

如何利用天赋实现最大化的价值输出

这种文章,以我现在的实力很难写出来。所以需要引用一些视频。 上92高校容易吗 如果基于天赋努力,非常容易。 如果不是这样,非常非常难。 高考失败人生完蛋?复读考上交大,进入社会才发现学历只是一张纸,98…

LigerUI在MVC模式下的响应原则

LigerUI是基于jQuery的UI框架,故他也是遵守jQuery的开发模式,但是也具有其特色的侦听函数,那么当LigerUI作为View层的时候,他所发送后端的必然是表单的数据,在此我们以俩个div为例: {Layout "~/View…

【力扣】49.字母异位词分组

AC截图 题目 思路 由于互为字母异位词的两个字符串包含的字母相同,因此对两个字符串分别进行排序之后得到的字符串一定是相同的,故可以将排序之后的字符串作为哈希表的键。 可以遍历strs,将其中每一个str排序,然后用unodered_ma…

docker安装nacos2.2.4详解(含:nacos容器启动参数、环境变量、常见问题整理)

一、镜像下载 1、在线下载 在一台能连外网的linux上执行docker镜像拉取命令 docker pull nacos:2.2.4 2、离线包下载 两种方式: 方式一: -)在一台能连外网的linux上安装docker执行第一步的命令下载镜像 -)导出 # 导出镜像到…

【图床配置】PicGO+Gitee方案

【图床配置】PicGOGitee方案 文章目录 【图床配置】PicGOGitee方案为啥要用图床图床是什么配置步骤下载安装PicGoPicGo配置创建Gitee仓库Typora中的设置 为啥要用图床 在Markdown中,图片默认是以路径的形式存在的,类似这样 可以看到这是本地路径&#x…

【C++】类与对象(下)

🦄 个人主页: 小米里的大麦-CSDN博客 🎏 所属专栏: 小米里的大麦——C专栏_CSDN博客 🎁 代码托管: 小米里的大麦的Gitee仓库 ⚙️ 操作环境: Visual Studio 2022 文章目录 1. 再谈构造函数1.1 构造函数体赋值1.2 初始化列表1.3 explicit 关键…

SpringBoot笔记

1.创建 使用idea提供的脚手架创建springboot项目,选上需要的模块,会自动进行导包 打成jar包,之前直接用原生的maven打包的是一个瘦jar,不能直接跑,把服务器上部署的jar排除在外了,但是现在加上打包查件&am…

Fiddler(一) - Fiddler简介_fiddler软件

文章目录 一、为什么选择Fiddler作为抓包工具? 二、什么是Fiddler?三、Fiddler使用界面简介四、延伸阅读 一、为什么选择Fiddler作为抓包工具? 抓包工具有很多,小到最常用的web调试工具firebug,大到通用性强大的抓包工具wireshark。为什么使用fid…

受击反馈HitReact、死亡效果Death Dissolve、Floating伤害值Text(末尾附 客户端RPC )

受击反馈HitReact 设置角色受击标签 (GameplayTag基本了解待补充) 角色监听标签并设置移动速度 创建一个受击技能,并应用GE 实现设置角色的受击蒙太奇动画 实现角色受击时播放蒙太奇动画,为了保证通用性,将其设置为一个函数,并…

Vue+Echarts 实现青岛自定义样式地图

一、效果 二、代码 <template><div class"chart-box"><chart ref"chartQingdao" style"width: 100%; height: 100%;" :options"options" autoresize></chart></div> </template> <script> …

线段树(Segment Tree)和树状数组

线段树&#xff08;Segment Tree&#xff09;和树状数组 线段树的实现链式&#xff1a;数组实现 解题思路树状数组 线段树是 二叉树结构 的衍生&#xff0c;用于高效解决区间查询和动态修改的问题&#xff0c;其中区间查询的时间复杂度为 O(logN)&#xff0c;动态修改单个元素的…

计算机网络 (62)移动通信的展望

一、技术发展趋势 6G技术的崛起 内生智能&#xff1a;6G将强调自适应网络架构&#xff0c;通过AI驱动的智能算法提升通信能力。例如&#xff0c;基于生成式AI的6G内生智能架构将成为重要研究方向&#xff0c;实现低延迟、高效率的智能通信。信息编码与调制技术&#xff1a;新型…

数据结构 前缀中缀后缀

目录 前言 一&#xff0c;前缀中缀后缀的基本概念 二&#xff0c;前缀与后缀表达式 三&#xff0c;使用栈实现后缀 四&#xff0c;由中缀到后缀 总结 前言 这里学习前缀中缀后缀为我们学习树和图做准备&#xff0c;这个主题主要是对于算术和逻辑表达式求值&#xff0c;这…