【数据结构】【栈】括号匹配算法(可直接运行)

article/2025/10/25 22:02:32

文章目录

  • 1.算法问题
  • 2.算法思路
  • 3.算法实现
  • 4.完整代码
  • 5.运行结果

1.算法问题

假设表达式中允许包含三种括号:()、[]、{},其嵌套的顺序任意,即([]{})或[()[]{}]等均为正确的格式,实现一个算法,完成对表达式中括号的检验。

2.算法思路

考虑以下括号序列:
[ ( [ ] [ ] ) ]

分析如下:
(1)计算机接收第一个括号 [ 后,等待与之匹配的 ] 出现。
(2)接收第二个括号 ( 后,第一个括号放置一边,等待与之匹配的 ) 出现。
(3)接收第三个括号 [ 后,第二个括号暂时放置一边,等待能与之匹配的 ] 出现。
(4)直至第四个括号出现,发现其能与最近的第三个括号匹配,那第三个括号匹配完成,退出等待。
(5)以此类推,第五个等待,第六个与第五个匹配完成后,第五个退出等待;第七个与第二个匹配完成后,第二个退出等待;第八个与第一个匹配完成后,扫描完成,且等待序列中没有了数据,匹配成功。

可以发现,此算法与栈的特性类似,后出现的左括号先匹配

综上所述,可以得出大致算法如下:
(1)初始创立一个空栈。
(2)逐个读入括号。
(3)如果是左括号,那么就压入栈中。
(4)如果是右括号,则先检查栈是否为空,如果栈为空,则表示无左括号与之匹配,匹配失败。如果栈不为空,则进入下一步。
(5)从栈顶取出元素,检测此左括号与读入右括号是否匹配,如果不匹配,则返回匹配失败。如果匹配成功,则回到第二步,直至不满足循环条件。
(6)检测栈是否为空,如果栈为空,则表示匹配完成后还有多余的括号,匹配失败。若栈为空,则返回匹配成功。

3.算法实现

// 算法实现——括号匹配
bool bracketCheck(char str[], int length) {SeqStack S;// 初始化栈InitStack(S);// 通过遍历从左往右扫描字符for(int i = 0;i < length;i++) {// 如果字符是左括号,则压入栈中if(str[i] == '(' || str[i] == '[' || str[i] == '{') {Push(S, str[i]);} else {// 如果不是左括号,且栈为空则表示匹配失败// 因为栈中即是压入的左括号,如果匹配到右括号时,栈中没有左括号与之匹配,自然就不合理,匹配失败if(StackEmpty(S)) {return false;}// 创建临时变量,用于存储栈中所弹出的字符,即左括号char topElem;// 弹出栈顶元素Pop(S, topElem);// 如果括号类型不匹配,则视为匹配失败if(str[i] == ')' && topElem != '(') {return false;}if(str[i] == ']' && topElem != '[') {return false;}if(str[i] == '}' && topElem != '{') {return false;}}}// 检索完所有括号后,还需检测栈是否为空,如果栈不为空,说明还有未匹配完成项,即匹配失败// 栈空则匹配成功return StackEmpty(S);
}

4.完整代码

#include<stdio.h>#define MaxSize 10typedef struct {    // 定义顺序栈int data[MaxSize];  // 静态数组存放栈中元素int top;    // 栈顶指针:指向目前栈顶元素的位置
} SeqStack;// 初始化
void InitStack(SeqStack &S) {// 初始化栈顶指针,由于在顺序栈中,0表示栈底部的位置// 所以初始应赋小于0的值,常用  -1S.top = -1;
}// 判断栈空
bool StackEmpty(SeqStack S) {// 空栈条件:栈顶指针等于-1!!!// 说明栈中没有任何元素if(S.top == -1) {return true;} else {return false;}
}// 进栈操作
// 类似于线性表的插入操作,即“增”操作
// 但由于栈的特殊性,只允许在栈顶进行操作,所以可称为进栈操作
// 即在栈顶增加元素
bool Push(SeqStack &S, char x) {// 判断栈是否已满,由于top是静态数组最后一个元素的下标(栈顶元素下标)// 所以只需要判断top值是否已经达到静态数组最大长度-1即可if(S.top == MaxSize - 1) {return false;}// top指针+1-->下标加一,指向数组后一位S.top = S.top + 1;// 新元素入栈S.data[S.top] = x;// 以上操作等价于S.data[++S.top] = x;return true;
}// 出栈操作
// 类似于线性表的删除操作,即“删”操作
// 但由于栈的特殊性,只允许在栈顶进行操作,所以可称为出栈操作
// 即在栈顶弹出元素
bool Pop(SeqStack &S, char &x) {// 栈顶指针等于-1,栈为空,不可删除if(S.top == -1) {return false;}// 注意:虽然在此处出栈操作栈顶指针减一,可视为删除成功// 但实际上数据仍然保留在存储单元中,只是从逻辑上删除了// 将栈顶元素赋给x,x为引用变量,可从外部获取弹出的数据x = S.data[S.top];// 栈顶指针减一S.top = S.top - 1;// 以上操作等价于x= S.data[S.top--]return true;
}// 算法实现——括号匹配
bool bracketCheck(char str[], int length) {SeqStack S;// 初始化栈InitStack(S);// 通过遍历从左往右扫描字符for(int i = 0;i < length;i++) {// 如果字符是左括号,则压入栈中if(str[i] == '(' || str[i] == '[' || str[i] == '{') {Push(S, str[i]);} else {// 如果不是左括号,且栈为空则表示匹配失败// 因为栈中即是压入的左括号,如果匹配到右括号时,栈中没有左括号与之匹配,自然就不合理,匹配失败if(StackEmpty(S)) {return false;}// 创建临时变量,用于存储栈中所弹出的字符,即左括号char topElem;// 弹出栈顶元素Pop(S, topElem);// 如果括号类型不匹配,则视为匹配失败if(str[i] == ')' && topElem != '(') {return false;}if(str[i] == ']' && topElem != '[') {return false;}if(str[i] == '}' && topElem != '{') {return false;}}}// 检索完所有括号后,还需检测栈是否为空,如果栈不为空,说明还有未匹配完成项,即匹配失败// 栈空则匹配成功return StackEmpty(S);
}int main() {char a[] = "[([][])]";if(bracketCheck(a, 8)) {printf("The match is successful");}
}

5.运行结果

在这里插入图片描述


http://chatgpt.dhexx.cn/article/MeHIL9KF.shtml

相关文章

CronTrigger(重要,非常强大)

&#xff0c;&#xff1a;表示或的关系 - &#xff1a;表示至的关系 * &#xff1a;表示每的关系 / &#xff1a;表示每的关系 # &#xff1a;表示第的关系 L &#xff1a;表示最后的关系 6 &#xff1a;表示星期五

CronTrigger定时任务

1、CronTrigger的作用&#xff1a;基于日历的作业调度器&#xff0c;而不是像SimpleTrigger那样精确指定间隔时间&#xff0c;比SimpleTrigger更常用。 2、Cron表达式 用于配置CronTrigger实例 是由7个子表达式组成的字符串&#xff0c;描述了时间表的详细信息 格式&#x…

Quartz(02) 两种Trigger(SimpleTrigger/CronTrigger)

上一章讲了Quartz 的几个核心概念(上一章节地址Quartz快速入门).接下来看看Quartz的两种Trigger. SimpleTrigger/CronTrigger. 简单的定时任务,可以采用SimpleTrigger,复杂的任务一般采用CronTrigger. 源码下载地址 1.SimpleTrigger   SimpleTrigger是Quartz中最简单Trig…

Quartz CronTrigger最完整触发时间配置说明

CronTrigger配置格式: 格式: [秒] [分] [小时] [日] [月] [周] [年] 序号说明 是否必填 允许填写的值允许的通配符 1 秒 是 0-59 , - * / 2 分 是 0-59 , - * / 3小时 是 0-23 , - * / 4 日 是 1-31 , - * ? / L W 5 月 是 1-12 or JAN-DEC , - * / 6 周 是 1-7 or SUN-…

Quartz-CronTrigger解读

文章目录 概述Cron表达式的组成cron表达式中的问号(?) 的含义1、当星期和日期都为*或数字时&#xff0c;报错2、当星期和日期都为?时&#xff0c;报错 时间格式特殊字符Cron 表达式样例CronTrigger 样例 1 – 时钟从 0 分钟开始&#xff0c;每 5 分钟执行一次CronTrigger 样例…

SimpleTrigger和CronTrigger配置说明和cornExpression表达式

CronTrigger基于日历和时间&#xff0c;SimpleTrigger基于时间。 每天8:32,12:32 执行一次&#xff0c;启动的时候执行一次&#xff1b; Java代码 <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/sprin…

【Java定时任务】浅谈CronTrigger的用法和在线Cron表达式生成网址

目录 0.CronTrigger的作用 1.Cron表达式 2.Cron表达式特殊字符意义对应表 0.CronTrigger的作用 CronTriggers往往比SimpleTrigger更有用&#xff0c;如果您需要基于日历的概念&#xff0c;而非SimpleTrigger完全指定的时间间隔&#xff0c;复发的发射工作的时间表。 CronTri…

CronTrigger cron表达式

Cron表达式 简单触发可以使用SimplTrigger实现类.功能类似timer 复杂触发可以使用CronTrigger实现类,内部利用cron表达式描述各种复杂的时间调度计划 举例 表示2022年9月3日凌晨4点的cron表达式 0 0 4 3 9 ? 2022 * 表示任何值,如果在分的字段上编写*,表示每分钟都会触发 …

crond和crontab详解

###一、crontab是什么&#xff1f;   1. 定时任务软件种类 at    适合仅执行一次就结束的调度命令&#xff0c;需要启动一个后端的atd服务。 crontab 需要启动一个服务crond才行&#xff0c;crond服务通过crontab命令实现 anacron 无法周期性执行&#xff0c;只能以天为…

C语言实现贪吃蛇(easyx)

1.大一写的贪吃蛇&#xff0c;有兴趣的可以自取&#xff0c;为了大家移植方便没有写头文件&#xff0c;直接复制就可以运行&#xff0c;编译环境是vs2022。 但是运行之前首先要进行两个配置&#xff0c;不然会报错。 1>将字符集改为“使用多字节字符” 2>因为我为游戏…

【C语言】贪吃蛇小游戏代码详解

贪吃蛇的规则大家应该都知道&#xff0c;简单来说就是吃到食物则蛇身增长&#xff0c;蛇可以在游戏区域内随意移动。游戏结束条件就是蛇撞到边界或者撞到自己色蛇身。 首先&#xff0c;实现一个函数&#xff0c;这个函数的作用是将光标移动到我们所期望的位置 /** 控制光标…

乐趣c语言——贪吃蛇

目录 ​ 1.游戏介绍与思路 1.1介绍&#xff1a; 1.2游戏区思路 1.2.1围墙 1.2.2游戏区域 1.2.3蛇头与蛇身 1.3游戏逻辑思路 2.正式实现贪吃蛇 2.1在开始前我们先包头文件和宏定义 2.2创建游戏区并且初始化 2.3初始化蛇头与蛇身 2.4打印与覆盖蛇 2.5实现蛇的移动 2…

C语言制作小游戏——贪吃蛇

直接上代码 ​ #include <stdio.h> #include <stdlib.h> #include <math.h> #include <conio.h> #include <time.h> #include <windows.h>//MAXWIDTH、MAXHEIGHT、INITLEN 以字符记 #define MAXWIDTH (30) #define MAXHEIGHT MAXWIDTH #d…

C语言之贪吃蛇游戏源码

贪吃蛇游戏是我编程以来做的第一个游戏&#xff0c;很兴奋&#xff0c;我记得当时我一直最求代码数量的最小化&#xff0c;即以最少的代码实现功能&#xff0c;有的时候&#xff0c;对换行、括号之类的都下手&#xff0c;呵呵&#xff0c;现在想想都想笑&#xff0c;那些年让我…

c语言贪吃蛇(简易版本含完整代码)

文章目录 前引头文件注释部分头文件部分主函数部分实现效果 前引 这个贪吃蛇是在我学完C语言的基本部分 在跟着哈工大的c语言程序设计Mooc课程完了之后做的 因为学完C语言很迷茫 不晓得应该学什么 学校里面的题库感觉刷起来又没有实际我能看到的东西 我认为在阶段性的学习过程中…

C语言实现简单贪吃蛇代码

基本构成 穿墙控制 死亡控制 初始化蛇&#xff08;init_snake函数&#xff09; 移动蛇&#xff08;move_snake函数&#xff09; 打印蛇&#xff08; print_snake函数&#xff09; 利用for循环检测坐标对应的蛇的位置 代码演示 #include <stdio.h> #include <window…

C语言实现贪吃蛇

C语言实现贪吃蛇 贪吃蛇项目是在b站的一位老师那里边学边敲的&#xff08;现在视频被老师删掉了&#xff09;&#xff0c;在这里记录一下代码和学习时的一些思考&#xff0c;希望自己以后能用链表将贪吃蛇实现。 思考过程&#xff1a; 贪吃蛇 C语言实现 1.打印首页 --FirstP…

C语言写个贪吃蛇游戏

贪吃蛇是个非常经典的游戏&#xff0c;用C语言来实现也是一个好玩的事情。这个游戏我写完后放在知乎&#xff0c;竟然点赞的人数超级多。我觉得大家喜欢&#xff0c;一个方面是因为写得简单&#xff0c;大家都能看得懂&#xff0c;一个可扩展性还是非常强的。 我试了说一下这个…

C语言实现贪吃蛇小游戏(详解)

目录 一、程序实现的原理&#xff1a; 二、预备知识 1、控制台窗口 2、通过代码将光标移动到控制台指定位置 三、 程序框架 四、具体操作 1、定义相关头文件 2、定义函数结构以及相关属性 五、完成各代码块 1、 实现将光标移到指定位置的操作 2、 打印游戏边框以及游戏初始…

小白入门——“贪吃蛇”的C语言实现(详细)

C语言实现&#xff0c;编译环境VS&#xff12;&#xff10;&#xff11;&#xff17; 附&#xff1a;easyx图形化(文章末尾&#xff09; 效果图如下 &#xff08;有一些函数kbhit&#xff0c;getch&#xff0c;在这表示为_kbhit与_getch&#xff09;//不同编译器原因 注意在Dev…