C语言实现括号匹配检查及栈的应用详解
目录
栈数据结构简介
C语言实现栈
栈的初始化
栈的销毁
栈的插入
栈的删除
栈的判空
获取栈顶数据
利用栈实现括号匹配检查
总结

在编程中,经常会遇到需要检查括号是否匹配的问题,比如在编译器中检查代码的语法正确性,或者在文本处理中确保括号的正确使用。本文将通过C语言实现一个括号匹配检查的程序,并详细介绍栈数据结构在其中的应用。
栈数据结构简介
栈是一种后进先出(LIFO, Last In First Out)的数据结构。它就像一个放盘子的栈,最后放上去的盘子总是最先被拿下来。在栈中,有几个基本操作:
- 初始化(Init):创建一个空栈。
- 销毁(Destroy):释放栈占用的内存。
- 插入(Push):将元素添加到栈顶。
- 删除(Pop):从栈顶移除元素。
- 判空(Empty):检查栈是否为空。
- 获取元素个数(Size):返回栈中元素的数量。
- 获取栈顶数据(Top):返回栈顶的元素,但不删除它。
C语言实现栈
c#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>typedef char typestk;
typedef struct Stack
{typestk* a;int top;int capacity;
} ST;
这里定义了一个栈的结构体 Stack ,包含一个字符类型的指针 a 用于存储栈中的元素, top 表示栈顶的位置, capacity 表示栈的容量。
栈的初始化
c// 栈的初始化
void STInit(ST* ps)
{assert(ps);ps->a = (typestk*)malloc(sizeof(typestk) * 4);if (ps->a == NULL){perror("malloc");return;}ps->top = 0;ps->capacity = 4;
}
STInit 函数用于初始化栈。首先通过 malloc 分配4个字符大小的内存空间给栈的数组 a ,如果分配失败,使用 perror 打印错误信息。然后将栈顶位置 top 设为0,栈容量 capacity 设为4。
栈的销毁
c// 栈的销毁
void STDestory(ST* ps)
{// 查空assert(ps);free(ps->a);ps->a = NULL;ps->capacity = ps->top = 0;
}
STDestory 函数用于销毁栈。先使用 free 释放栈数组 a 占用的内存,然后将指针 a 设为 NULL ,并将栈顶位置 top 和栈容量 capacity 都设为0。
栈的插入
c// 栈的插入
void STPush(ST* ps, typestk x)
{assert(ps);if (ps->top == ps->capacity){typestk* tmp = (typestk*)realloc(ps->a, sizeof(typestk) * ps->capacity * 2);if (tmp == NULL){perror("realloc error");return;}ps->a = tmp;ps->capacity *= 2;}ps->a[ps->top] = x;ps->top++;
}
STPush 函数用于将元素插入栈顶。首先检查栈是否已满(即 top 等于 capacity ),如果已满,使用 realloc 将栈的容量扩大为原来的2倍。如果内存重新分配失败,打印错误信息并返回。然后将元素 x 插入到栈顶位置,并将栈顶位置 top 加1。
栈的删除
c// 栈的删除
void STPop(ST* ps)
{assert(ps);assert(!STEmpty(ps)); ps->top--;
}
STPop 函数用于从栈顶删除元素。首先检查栈是否为空,如果为空则使用 assert 断言报错。然后将栈顶位置 top 减1,相当于删除了栈顶元素。
栈的判空
c// 栈的判空
bool STEmpty(ST* ps)
{assert(ps);return ps->top == 0;
}STEmpty 函数用于检查栈是否为空。如果栈顶位置 top 为0,则返回 true ,否则返回 false 。栈的元素个数c// 栈的元素个数
int STSize(ST* ps)
{assert(ps);return ps->top;
}
STSize 函数返回栈中元素的个数,即栈顶位置 top 的值。
获取栈顶数据
c// 获取栈顶数据
typestk STTop(ST* ps)
{assert(ps);assert(!STEmpty(ps));return ps->a[ps->top - 1];
}
STTop 函数返回栈顶的元素,但不删除它。首先检查栈是否为空,如果为空则使用 assert 断言报错。然后返回栈顶位置 top - 1 处的元素。
利用栈实现括号匹配检查
cbool isValid(char* s)
{ST st;STInit(&st);while (*s){if (*s == '[' || *s == '(' || *s == '{'){STPush(&st, *s);}else{if (STEmpty(&st)){STDestory(&st);return false;}char top = STTop(&st);STPop(&st);if ((*s == ')' && top != '(') || (*s == ']' && top != '[') || (*s == '}' && top != '{')){STDestory(&st);return false;}}++s;}bool ret = STEmpty(&st);STDestory(&st);return ret;
}
isValid 函数用于检查给定字符串中的括号是否匹配。
初始化一个栈 st 。
遍历字符串 s :
- 如果当前字符是左括号( [ 、 ( 、 { ),将其压入栈中。
- 如果当前字符是右括号( ] 、 ) 、 } ):
- 检查栈是否为空,如果为空,说明没有匹配的左括号,返回 false 。
- 否则,获取栈顶元素并弹出栈,检查栈顶元素是否与当前右括号匹配。如果不匹配,返回 false 。
遍历结束后,如果栈为空,说明所有括号都匹配,返回 true ;否则返回 false 。最后销毁栈。

总结
通过以上代码,我们实现了一个简单的栈数据结构,并利用栈解决了括号匹配检查的问题。栈作为一种基本的数据结构,在很多算法和应用中都有广泛的应用,如表达式求值、深度优先搜索等。掌握栈的基本操作和应用场景,对于提升编程能力和解决实际问题都非常有帮助。希望本文能帮助你更好地理解栈和括号匹配问题。
相关文章:
C语言实现括号匹配检查及栈的应用详解
目录 栈数据结构简介 C语言实现栈 栈的初始化 栈的销毁 栈的插入 栈的删除 栈的判空 获取栈顶数据 利用栈实现括号匹配检查 总结 在编程中,经常会遇到需要检查括号是否匹配的问题,比如在编译器中检查代码的语法正确性,或者在…...
C语言中的字符串与数组的关系
在C语言中,字符串和数组之间有着紧密的关系。理解它们的区别和联系对于编写高效且可靠的代码至关重要。在本篇博文中,我们将详细分析字符串和数组在C语言中的概念、它们的关系以及如何在编程中应用它们。 一、字符串与数组的基础知识 1.1 数组概念 在C语言中,数组是一组相…...
阿里云魔笔低代码应用开发平台快速搭建教程
AI低代码,大模型时代应用开发新范式 什么是魔笔 介绍什么是魔笔低代码应用开发平台。 魔笔是一款面向全端(Web、H5、全平台小程序、App)场景的模型驱动低代码开发平台,提供一站式的应用全生命周期管理,包括可视化开发…...
A Survey on Mixture of Experts 混合专家模型综述(第二部分:混合专家系统设计)
A Survey on Mixture of Experts 混合专家模型综述 (第一部分:混合专家算法设计) A Survey on Mixture of Experts arxiv github:A-Survey-on-Mixture-of-Experts-in-LLMs 5 System Design of Mixture of Experts While Mixture of Exper…...
docker python:latest镜像 允许ssh远程
跳转到家目录 cd创建pythonsshdockerfile mkdir pythonsshdockerfile跳转pythonsshdockerfile cd pythonsshdockerfile创建Dockerfile文件 vim Dockerfile将Dockerfile的指令复制到文件中 # 使用 python:latest 作为基础镜像 # 如果我的镜像列表中没有python:latest镜像&…...
通过 CSS 的 命名页面(Named Pages) 技术实现作用域隔离,实现 @page 样式仅影响当前组件
以下是实现 page 样式仅影响当前组件的完整解决方案,通过 CSS 的 命名页面(Named Pages) 技术实现作用域隔离: vue <template><div><button v-print"printOptions">打印当前报表</button><…...
Aim Robotics电动胶枪:机器人涂胶点胶的高效解决方案
在自动化和智能制造领域,机器人技术的应用越来越广泛,而涂胶和点胶作为生产过程中的重要环节,也逐渐实现了自动化和智能化。Aim Robotics作为一家专注于机器人技术的公司,其推出的电动胶枪为这一领域带来了高效、灵活且易于操作的…...
动态规划----完全平方数(3种写法,逐步简化)
题目链接:完全平方数 完全平方数可以认为是完全背包问题。每一个平方小于n的平方数都是物品,而完全平方数之和n就是背包容量。每一个平方和都可以无限次使用。 写法1:把所有小于n的平方数存入数组nums,使用二维dp数组。 递推公式的推导可以…...
C#中通过Response.Headers设置自定义参数
一、基础设置方法 1. 直接添加自定义头 // ASP.NET Core方案 Response.Headers.Append("X-API-Version", "2.3.1"); Response.Headers.Append("Custom-Auth-Token", Guid.NewGuid().ToString());• 底层原理:通过IHeaderDictionary…...
【HDLbits--分支预测器简单实现】
HDLbits--分支预测器简单实现 1 timer2.branche predicitors3.Branch history shift4.Branch direction predictor 以下是分支预测器的简单其实现; 1 timer 实现一个计时器,当load1’b1时,加载data进去,当load1’b0时进行倒计时&…...
LLM自动化评测
使用的数据集:ceval-exam import requests from datasets import load_dataset, concatenate_datasets import re from tqdm import tqdm import re, time, tiktoken, ollama from ollama import ChatResponse from ollama import Optionsdef llm(model, query, te…...
Linux--操作系统/进程
ok,我们今天学习linux中的操作系统和进程 1. 冯诺依曼体系 我们常⻅的计算机,如笔记本。我们不常⻅的计算机,如服务器,⼤部分都遵守冯诺依曼体系。 内存是CPU和外设之间的一个巨大的缓存! 截⾄⽬前,我们…...
MFC控件按钮的使用
MFC窗口的创建/消息映射机制 mfc.h #include<afxwin.h>//mfc头文件//应用程序类 class MyApp:public CWinApp //继承于应用程序类 { public://程序入口virtual BOOL InitInstance(); };//框架类 class MyFrame:public CFrameWnd { public:MyFrame();//声明宏 提供消息映…...
Java面试八股—Redis篇
一、Redis的使用场景 (一)缓存 1.Redis使用场景缓存 场景:缓存热点数据(如用户信息、商品详情),减少数据库访问压力,提升响应速度。 2.缓存穿透 正常的访问是:根据ID查询文章&…...
计算矩阵边缘元素之和(信息学奥赛一本通-1121)
【题目描述】 输入一个整数矩阵,计算位于矩阵边缘的元素之和。所谓矩阵边缘的元素,就是第一行和最后一行的元素以及第一列和最后一列的元素。 【输入】 第一行分别为矩阵的行数m和列数n(m<100,n<100),…...
Web后端开发之Maven
Maven Mven是apache旗下的一个开源项目,用来管理和构建java项目的工具。 通过一小段描述信息来管理项目。 Maven的作用 1.依赖管理:方便快捷的管理项目依赖的资源(jar包),避免版本冲突问题 以前用某个jar包需要下载…...
哈希算法,蓝桥杯java备战中
哈希表的实现 核心思路 目标:实现一个基于开放寻址法(线性探测)的哈希表,支持插入元素 I x 和查询元素 Q x 两种操作。 核心逻辑: 哈希函数:将元素映射到固定范围的索引(哈希值)。…...
there are no enabled repos
我做了两个操作 第一个操作: 1.先在本地电脑,也就是在我们电脑的桌面上下载 https://repo.huaweicloud.com/repository/conf/CentOS-7-reg.repo 2.在CentOS 创建etc文件夹 3在etc文件夹内创建yum.repos.d文件夹 4.将下载好的repo 黏贴到yum.repos.d…...
OpenEuler-22.03-LTS上利用Ansible轻松部署MySQL 5.7
一、需求 使用ansible自动化部署mysql二进制部署mysql部署mysql并创建JDBC用户 二、环境信息 本文涉及的代码,配置文件地址: 链接:百度网盘 请输入提取码 提取码:1g6y 软件名称版本备注Ansible2.9.27All modules — Ansible Doc…...
前端无限滚动内容自动回收技术详解:原理、实现与优化
文章目录 一、核心需求与技术挑战1.1 无限滚动的问题症结1.2 自动回收的三大目标 二、技术实现原理2.1 虚拟滚动核心机制2.2 关键技术指标 三、完整实现方案3.1 基础HTML结构3.2 CSS关键样式3.3 JavaScript核心逻辑3.3.1 滚动控制器3.3.2 动态尺寸处理 四、性能优化策略4.1 内存…...
LeetCode hot 100 每日一题(9)——560. 和为 K 的子数组
这是一道难度为中等的题目,让我们来看看题目描述: 给你一个整数数组 nums 和一个整数 k ,请你统计并返回 该数组中和为 k 的子数组的个数 。 子数组是数组中元素的连续非空序列。 示例 1: 输入: nums [1,1,1], k 2 输…...
C++Primer学习(6.7 函数指针——难!)
6.7 函数指针 (这一章节比较难) 函数指针指向的是函数而非对象。和其他指针一样,函数指针指向某种特定类型。函数的类型由它的返回类型和形参类型共同决定,与函数名无关。例如: //比较两个 string 对象的长度 bool lengthCompare(const string &,co…...
单一责任原则在Java设计模式中的深度解析
在软件开发中,设计模式提供了一种解决特定问题的思路。在众多的设计原则中,单一责任原则(Single Responsibility Principle,SRP)是一个非常重要的概念。它主要强调一个类应该只有一个责任,也就是说…...
如何在Ubuntu上构建编译LLVM和ISPC,以及Ubuntu上ISPC的使用方法
之前一直在 Mac 上使用 ISPC,奈何核心/线程太少了。最近想在 Ubuntu 上搞搞,但是 snap 安装的 ISPC不知道为什么只能单核,很奇怪,就想着编译一下,需要 Clang 和 LLVM。但是 Ubuntu 很搞,他的很多软件版本是…...
学习计划:第四阶段(第十周)
目录 第四阶段:特殊方法与高级特性 第 10 周:综合复习与实践 周一 周二 周三 周四 周五 总结 一、项目设计与实现 二、问题与解决 三、学习成果 四、后续展望 第四阶段:特殊方法与高级特性 第 10 周:综合复习与实践 …...
如何查看redis的缓存时间
要查看 Redis 缓存的时间,有下列两种方式: 一、使用 TTL 命令来获取缓存剩余的时间 Redis提供了多个命令来查看缓存数据的时间戳,其中最常用的命令是ttl和pttl。 ttl它返回的是以秒为单位的时间,表示 key 距离过期的时间还有多久…...
每日学习Java之一万个为什么
JVM的加载过程 启动阶段:启动JVM实例,设置初始配置参数,加载核心类库如java.lang类加载器:自举加载器,扩展加载器,系统加载器,自定义加载器。分别负责- 1.核心类库rt.jar等 2.扩展目录下的类库…...
【MySQL】表的约束(上)
文章目录 表的约束什么是表的约束空属性默认值列描述(comment)零填充(zerofill)主键 总结 表的约束 什么是表的约束 表的约束(Constraints)是数据库表中的规则,用于限制存储的数据,…...
静态分析技术:Jadx-GUI高级用法与模式识别
1. 深度反编译策略 1.1 多层级反混淆方案 代码恢复流程: graph TD A[混淆代码] --> B{符号恢复} B -->|字典匹配| C[变量重命名] B -->|类型推导| D[参数重构] C --> E[控制流优化] D --> E E --> F[语义化输出] 反混淆脚本示例&…...
30天学习Java第六天——Object类
Object类 java.lang.Object时所有类的超类。Java中所有类都实现了这个类中的方法。 toString方法 将Java对象转换成字符串的表示形式。 public String toString() {return getClass().getName() "" Integer.toHexString(hashCode()); }默认实现是:完…...
