C语言实现括号匹配检查及栈的应用详解

news2025/4/22 20:35:43

目录

栈数据结构简介 

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 处的元素。
 


利用栈实现括号匹配检查

 
c
  
bool 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 。最后销毁栈。
 


总结
 


通过以上代码,我们实现了一个简单的栈数据结构,并利用栈解决了括号匹配检查的问题。栈作为一种基本的数据结构,在很多算法和应用中都有广泛的应用,如表达式求值、深度优先搜索等。掌握栈的基本操作和应用场景,对于提升编程能力和解决实际问题都非常有帮助。希望本文能帮助你更好地理解栈和括号匹配问题。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

阿里云魔笔低代码应用开发平台快速搭建教程

AI低代码&#xff0c;大模型时代应用开发新范式 什么是魔笔 介绍什么是魔笔低代码应用开发平台。 魔笔是一款面向全端&#xff08;Web、H5、全平台小程序、App&#xff09;场景的模型驱动低代码开发平台&#xff0c;提供一站式的应用全生命周期管理&#xff0c;包括可视化开发…

A Survey on Mixture of Experts 混合专家模型综述(第二部分:混合专家系统设计)

A Survey on Mixture of Experts 混合专家模型综述 (第一部分&#xff1a;混合专家算法设计) A Survey on Mixture of Experts arxiv github&#xff1a;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镜像&…

Aim Robotics电动胶枪:机器人涂胶点胶的高效解决方案

在自动化和智能制造领域&#xff0c;机器人技术的应用越来越广泛&#xff0c;而涂胶和点胶作为生产过程中的重要环节&#xff0c;也逐渐实现了自动化和智能化。Aim Robotics作为一家专注于机器人技术的公司&#xff0c;其推出的电动胶枪为这一领域带来了高效、灵活且易于操作的…

【HDLbits--分支预测器简单实现】

HDLbits--分支预测器简单实现 1 timer2.branche predicitors3.Branch history shift4.Branch direction predictor 以下是分支预测器的简单其实现&#xff1b; 1 timer 实现一个计时器&#xff0c;当load1’b1时&#xff0c;加载data进去&#xff0c;当load1’b0时进行倒计时&…

Linux--操作系统/进程

ok&#xff0c;我们今天学习linux中的操作系统和进程 1. 冯诺依曼体系 我们常⻅的计算机&#xff0c;如笔记本。我们不常⻅的计算机&#xff0c;如服务器&#xff0c;⼤部分都遵守冯诺依曼体系。 内存是CPU和外设之间的一个巨大的缓存&#xff01; 截⾄⽬前&#xff0c;我们…

Java面试八股—Redis篇

一、Redis的使用场景 &#xff08;一&#xff09;缓存 1.Redis使用场景缓存 场景&#xff1a;缓存热点数据&#xff08;如用户信息、商品详情&#xff09;&#xff0c;减少数据库访问压力&#xff0c;提升响应速度。 2.缓存穿透 正常的访问是&#xff1a;根据ID查询文章&…

Web后端开发之Maven

Maven Mven是apache旗下的一个开源项目&#xff0c;用来管理和构建java项目的工具。 通过一小段描述信息来管理项目。 Maven的作用 1.依赖管理&#xff1a;方便快捷的管理项目依赖的资源&#xff08;jar包&#xff09;&#xff0c;避免版本冲突问题 以前用某个jar包需要下载…

there are no enabled repos

我做了两个操作 第一个操作&#xff1a; 1.先在本地电脑&#xff0c;也就是在我们电脑的桌面上下载 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用户 二、环境信息 本文涉及的代码&#xff0c;配置文件地址&#xff1a; 链接&#xff1a;百度网盘 请输入提取码 提取码&#xff1a;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 内存…

如何在Ubuntu上构建编译LLVM和ISPC,以及Ubuntu上ISPC的使用方法

之前一直在 Mac 上使用 ISPC&#xff0c;奈何核心/线程太少了。最近想在 Ubuntu 上搞搞&#xff0c;但是 snap 安装的 ISPC不知道为什么只能单核&#xff0c;很奇怪&#xff0c;就想着编译一下&#xff0c;需要 Clang 和 LLVM。但是 Ubuntu 很搞&#xff0c;他的很多软件版本是…

【MySQL】表的约束(上)

文章目录 表的约束什么是表的约束空属性默认值列描述&#xff08;comment&#xff09;零填充&#xff08;zerofill&#xff09;主键 总结 表的约束 什么是表的约束 表的约束&#xff08;Constraints&#xff09;是数据库表中的规则&#xff0c;用于限制存储的数据&#xff0c…

静态分析技术:Jadx-GUI高级用法与模式识别

1. 深度反编译策略 1.1 多层级反混淆方案 代码恢复流程&#xff1a; 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()); }默认实现是&#xff1a;完…

【C语言】编译和链接详解

hi&#xff0c;各位&#xff0c;让我们开启今日份博客~ 小编个人主页点这里~ 目录 一、翻译环境和运行环境1、翻译环境1.1预处理&#xff08;预编译&#xff09;1.2编译1.2.1词法分析1.2.2语法分析1.2.3语义分析 1.3汇编1.4链接 2.运行环境 一、翻译环境和运行环境 在ANSI C…

DataWhale 速通AI编程开发:(基础篇)第1章 环境下载、安装与配置

课程地址&#xff1a;Datawhale-学用 AI,从此开始 vscode 更新为最新版 目前绝大多数deepseek非官方渠道均兼容openai的api格式&#xff0c;这里以硅基流动为例进行演示&#xff0c;其他非官方渠道同理。 点击链接注册账号之后&#xff0c;点击“实名认证“完成实名&#xff0…

本地知识库RAG总结

目录 RAG流程: 知识库的要求&#xff1a; 知识抽取&#xff1a; 知识存储: 向量化: 知识检索: 应用客户端: RAG智能问答应用几个痛点&#xff1a; 如何提升召回率改进思路&#xff1a; 如何提升回答专业性&#xff1a; RAG评测&#xff1a; 总结&#xff1a; 参考…

torch_geometric 安装

环境监测&#xff1a; import torch print(torch.__version__) # 查看pytorch安装的版本号 print(torch.cuda.is_available()) # 查看cuda是否可用。True为可用&#xff0c;即是gpu版本pytorch print(torch.cuda.get_device_name(0)) # 返回GPU型号 …

网页打印很简单!用web打印插件lodop轻松实现文件打印

最近&#xff0c;给客户发一个事件提醒软件&#xff0c;其中客户要求实现打印功能&#xff0c;因为是用asp.net mvc 开发首先考虑到用水晶报表来实现&#xff08;crystalReport&#xff09;&#xff0c;以前开发c# winform程序&#xff0c;感觉水晶报表还是蛮好的&#xff0c;但…