括号匹配问题详解

article/2025/10/25 19:11:34

直接看题!!

typedef char STDataType;typedef struct Stack
{STDataType* a;int top;   //栈顶的位置int capacity; //容量
}ST;void StackInit(ST* ps)
{assert(ps);ps->a = NULL;ps->top = 0;ps->capacity = 0;}void StackDestroy(ST* ps)
{assert(ps);free(ps->a);ps->a = NULL;ps->capacity = ps->top = 0;
}void StackPush(ST* ps, STDataType x)
{assert(ps);//容量不足,需要进行扩容if (ps->top == ps->capacity){//增大容量int NewCapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;//对a指向的数组空间进行增容ps->a = (STDataType*)realloc(ps->a, NewCapacity * sizeof(STDataType));//检查if (ps->a == NULL){printf("realloc fail\n");exit(-1);}ps->capacity = NewCapacity;}//增容之后再入栈//top是从0位置开始所以是先使用再++ps->a[ps->top] = x;ps->top++;
}void StackPop(ST* ps)
{assert(ps);assert(ps->top > 0);--ps->top;}bool StackEmpty(ST* ps)
{assert(ps);if (ps->top > 0){return false;}else{return true;}//return ps->top == 0;
}int StackSize(ST* ps)
{assert(ps);return ps->top;
}STDataType StackTop(ST* ps)
{assert(ps);assert(ps->top > 0);//top是从0开始的return ps->a[ps->top - 1];}

  这道题不能用直接数括号数量来进行解答,因为还涉及顺序的问题。括号的匹配用什么来做是最合适的呢?答案是用栈。

  根据栈独有的特点,具体操作:1、属于左括号,进行入栈处理。2、属于右括号进行出栈处理,然后进行匹配,不匹配就报错。我们既然选择用C语言来实现,就需要我们自己提前实现一个栈结构。代码如下:

在栈的基础上再进行括号匹配的实现,代码如下:


bool isValid(char * s)
{ST st;//初始化栈StackInit(&st);while(*s){//如果是左括号就入栈if(*s == '(' || *s == '[' || *s == '{'){//入栈StackPush(&st, *s);++s;}else{//若到了这里*s一定是右括号//若只有右括号,栈为空if(StackEmpty(&st)){return false;}//取出栈顶元素char top = StackTop(&st);//出栈StackPop(&st);//这里不要判断相等,判断相等是不能立即得出结果的,括号有三个种类,//就算第一种匹配了,还要判断第二种或者第三种是否匹配//所以这里应判断不等,一旦这种括号不相等就立即能得到结果if((*s == ')' && top != '(')|| (*s == ']' && top != '[')|| (*s == '}' && top != '{')){StackDestroy(&st);//但凡有不匹配都会return falsereturn false;}else{//到这里说明匹配上了,继续比++s;}}}//栈为空,说明所有左括号都匹配了,反之,不为空,返回falsebool tmp = StackEmpty(&st);StackDestroy(&st);//到这里就说明全部都匹配上了return tmp;
}

​​​​​​​

到目前为止我们所实现出来的代码,逻辑上貌似没有什么问题,但是当我们进行测试的时候,出现了有测试用例没有通过。说明我们的程序还是存在问题的。这就需要我们根据测试用例来对程序进行分析了,一般的方法是根据显示出的不通过的测试用例带入到程序中,然后自己根据程序的逻辑走一遍,相当于执行一遍程序,这样很大可能会发现程序的问题所在。

根据上图的信息我们得知输入的是“[”,也就是*s为‘[’,进入while语句,紧接着进入if语句,然后进行入栈,++s,此时的*s为 '\0',跳出while循环,最后执行return true;所以才会输出true。但是只有一个括号是不可能匹配的,应该输出false才对。知道了病因,我们就可以对症下药了。具体怎么解决呢?既然他只有一个左括号,但是会跳出while语句,因此我们可以再while语句下面做文章。如果括号都匹配上了,那栈里面应该是没有元素的,所以我们可以在while语句执行完成之后再对进行判空,栈为空就说明都匹配上了,如果栈里面还有落单得到括号。那就肯定没有匹配上。

​​​​​​​

当我们修改之后,再进行测试,发现还是有执行错误,根据测试用例可以看到,他这次给的是“]”,我们还是按照上面说的方法进行操作。会发现它给的错误信息显示StackTop这个函数的ps->top > 0 是有问题的,再看看一下此函的具体实现,显而易见,是assert语句给出的错误提示,若没有assert的话这里就是越界错误,你不知道在哪里发生的越界,更不容易让人发现。这里也体现出使用assert好处可见一斑。

再分析:输入的是右括号,走的是while循环里的else语句,它里面会去取栈顶的元素,但是目前栈里面是没有元素的,不符合assert的判定条件,因此会报断言错误。

最后这里还有小细节,在return 之前,我们最好将栈进行销毁,不然就会有内存泄露,但是一般内存泄露是检查不出来的。但是咱们得养成良好的习惯!


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

相关文章

括号匹配

题目 给定一个只包括 (, ), {, }, [, ] 的字符串,判断字符串是否有效。 有效字符串需满足: 左括号必须用相同类型的右括号闭合。 左括号必须以正确的顺序闭合。 注意空字符串可被认为是有效…

数据结构——括号匹配问题

括号匹配 给定一个字符串,其中的字符只包含三种括号:花括号{ }、中括号[ ]、圆括号( ),即它仅由 “( ) [ ] { }” 这六个字符组成。设计算法,判断该字符串是否有效,即字符串中括号是否匹配。括号匹配要求括号必须以正…

C语言详解括号匹配问题(栈的应用 )

文章目录 问题概述算法思路不匹配的情况实现流程图C语言代码结果测试 问题概述 检测括号是否成对出现 最后出现的左括号最先匹配(LIFO),和栈的后进先出异曲同工 每出现一个右括号,就抵消(出栈操作)掉一个…

括号匹配(c++)

20. 有效的括号(力扣) 给定一个只包括 (,),{,},[,] 的字符串 s ,判断字符串是否有效。 有效字符串需满足: 左括号必须用相同类型的右括号闭合。 左括号必须以正确的顺序闭…

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

文章目录 1.算法问题2.算法思路3.算法实现4.完整代码5.运行结果 1.算法问题 假设表达式中允许包含三种括号:()、[]、{},其嵌套的顺序任意,即([]{})或[()[]{}]等均为正确的格式,实现一个算法,完成对表达式中括号的检验…

CronTrigger(重要,非常强大)

,:表示或的关系 - :表示至的关系 * :表示每的关系 / :表示每的关系 # :表示第的关系 L :表示最后的关系 6 :表示星期五

CronTrigger定时任务

1、CronTrigger的作用:基于日历的作业调度器,而不是像SimpleTrigger那样精确指定间隔时间,比SimpleTrigger更常用。 2、Cron表达式 用于配置CronTrigger实例 是由7个子表达式组成的字符串,描述了时间表的详细信息 格式&#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、当星期和日期都为*或数字时,报错2、当星期和日期都为?时,报错 时间格式特殊字符Cron 表达式样例CronTrigger 样例 1 – 时钟从 0 分钟开始,每 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语言很迷茫 不晓得应该学什么 学校里面的题库感觉刷起来又没有实际我能看到的东西 我认为在阶段性的学习过程中…