c/c++蓝桥杯经典编程题100道(18)括号匹配

ops/2025/2/13 16:00:59/

括号匹配

->返回c/c++蓝桥杯经典编程题100道-目录


目录

括号匹配

一、题型解释

二、例题问题描述

三、C语言实现

解法1:栈匹配法(难度★)

解法2:计数器法(仅限单一括号类型,难度★☆)

四、C++实现

解法1:使用STL栈(难度★)

解法2:哈希表优化(难度★★)

五、总结对比表

六、特殊方法与内置函数补充

1. C语言中的动态栈

2. C++的 unordered_map

3. 递归解法(不推荐)


一、题型解释

括号匹配是验证字符串中的括号是否正确闭合的问题。常见题型:

  1. 基础匹配:验证字符串中的 ()[]{} 是否成对且顺序正确。

  2. 包含其他字符:字符串中混合其他字符(如字母、数字),需跳过非括号字符。

  3. 最长有效括号子串:找到字符串中最长的有效括号子串长度。

  4. 最小添加次数:计算使括号匹配所需最少添加的括号数量。


二、例题问题描述

例题1:输入 "()[]{}",输出 true(所有括号正确匹配)。
例题2:输入 "{[()]}",输出 true(嵌套括号正确匹配)。
例题3:输入 "(]",输出 false(括号类型不匹配)。
例题4:输入 "()(()",输出最长有效子串长度 2


三、C语言实现

解法1:栈匹配法(难度★)

通俗解释

  • 像叠盘子一样,遇到左括号“放入盘子”,遇到右括号时检查最上面的盘子是否匹配。

c

#include <stdio.h>
#include <stdbool.h>
#include <string.h>#define MAX_SIZE 10000bool isValid(char *s) {char stack[MAX_SIZE]; // 用数组模拟栈int top = -1;         // 栈顶指针for (int i = 0; s[i] != '\0'; i++) {char c = s[i];if (c == '(' || c == '[' || c == '{') { // 左括号入栈stack[++top] = c;} else { // 右括号检查匹配if (top == -1) return false; // 栈为空,无法匹配char topChar = stack[top--];if ((c == ')' && topChar != '(') || (c == ']' && topChar != '[') || (c == '}' && topChar != '{')) {return false;}}}return top == -1; // 栈必须为空才完全匹配
}int main() {printf("%d\n", isValid("()[]{}")); // 输出 1(true)printf("%d\n", isValid("([)]"));    // 输出 0(false)return 0;
}

代码逻辑

  1. 栈初始化:数组 stack 模拟栈,top 表示栈顶索引。

  2. 遍历字符

    • 左括号:入栈,top 加1。

    • 右括号:若栈为空直接失败;否则弹出栈顶元素检查是否匹配。

  3. 最终检查:遍历结束后栈必须为空,否则存在未闭合的左括号。


解法2:计数器法(仅限单一括号类型,难度★☆)

通俗解释

  • 统计左右括号数量,左括号加1,右括号减1,中途不能为负,最终必须归零(仅适用于单一括号类型,如纯 ())。

c

bool isValidSingleType(char *s) {int balance = 0;for (int i = 0; s[i] != '\0'; i++) {if (s[i] == '(') balance++;else if (s[i] == ')') balance--;if (balance < 0) return false; // 中途右括号过多}return balance == 0;
}int main() {printf("%d\n", isValidSingleType("(()())")); // 输出 1printf("%d\n", isValidSingleType("())"));    // 输出 0return 0;
}

代码逻辑

  1. 平衡计数器:遇到左括号加1,右括号减1。

  2. 中途检查:计数器不能为负(右括号比左括号多)。

  3. 最终检查:计数器归零。

  4. 局限性:无法处理混合括号类型(如 [))。


四、C++实现

解法1:使用STL栈(难度★)

通俗解释

  • 使用C++标准库的 stack 容器,简化栈操作。

cpp

#include <iostream>
#include <stack>
using namespace std;bool isValid(string s) {stack<char> st;for (char c : s) {if (c == '(' || c == '[' || c == '{') { // 左括号入栈st.push(c);} else {if (st.empty()) return false; // 栈为空,无法匹配char topChar = st.top();st.pop();if ((c == ')' && topChar != '(') || (c == ']' && topChar != '[') || (c == '}' && topChar != '{')) {return false;}}}return st.empty(); // 栈必须为空
}int main() {cout << boolalpha; // 输出true/false而非1/0cout << isValid("{[()]}") << endl; // 输出 truecout << isValid("([)]") << endl;   // 输出 falsereturn 0;
}

代码逻辑

  • 与C语言栈方法逻辑相同,但使用 stack<char> 简化栈操作。


解法2:哈希表优化(难度★★)

通俗解释

  • 使用哈希表存储括号对,简化条件判断。

cpp

#include <iostream>
#include <stack>
#include <unordered_map>
using namespace std;bool isValidHash(string s) {stack<char> st;unordered_map<char, char> pairs = {{')', '('}, {']', '['}, {'}', '{'}};for (char c : s) {if (pairs.find(c) == pairs.end()) { // 左括号入栈st.push(c);} else { // 右括号检查匹配if (st.empty() || st.top() != pairs[c]) return false;st.pop();}}return st.empty();
}int main() {cout << isValidHash("()[]{}") << endl; // 输出 truereturn 0;
}

代码逻辑

  1. 哈希表存储映射:键为右括号,值为对应的左括号。

  2. 简化条件判断:遇到右括号时,直接从哈希表取对应的左括号进行比较。


五、总结对比表

方法时间复杂度空间复杂度优点缺点
栈匹配法O(n)O(n)支持所有括号类型需要额外空间
计数器法O(n)O(1)空间高效仅支持单一括号类型
STL栈O(n)O(n)代码简洁依赖STL库
哈希表优化O(n)O(n)代码可读性高需理解哈希表

六、特殊方法与内置函数补充

1. C语言中的动态栈

  • 实现方式:使用动态数组 (malloc 和 realloc) 实现栈,避免固定大小限制。

2. C++的 unordered_map

  • 作用:快速查找键值对,用于括号映射关系。

3. 递归解法(不推荐)

  • 局限性:递归深度与括号嵌套层数相关,可能导致栈溢出。

->返回c/c++蓝桥杯经典编程题100道-目录


http://www.ppmy.cn/ops/158080.html

相关文章

Java面试题(11) 整理Java面试题及参考答案

下面是最近翻阅各类博客网站收集整理的一些Java面试题&#xff0c;您值得拥有&#xff1a; 史上最全Java面试题&#xff08;带全部答案&#xff09; 2018JAVA面试题附答案(长期更新) 这是我见过最有用的java面试题&#xff0c;面试了无数公司总结的 2017 最新java面试题&…

android 自定义文件名和日期——android 打包技巧——不覆盖历史成功文件和版本-默认打包缺陷

一、传统方式 传统方式打包在 文件夹”app\release“下生成”app-release.apk“ 1. 多应用易混淆问题 同一项目多变体场景 当项目存在多个不同的构建变体时&#xff0c;例如不同的渠道包&#xff08;如应用宝渠道、华为应用市场渠道等&#xff09;、不同的版本类型&#xff08;…

网络安全之攻防笔记--web安全攻防

通用漏洞SQL SQL注入 针对数据库的攻击手法&#xff0c;通过在输入字段中插入恶意的SQL代码&#xff0c;改变或破坏原本预期的SQL注入查询 基于注入参数类型 数字型注入、字符型注入 基于请求提交方式 GET注入、POST注入 基于获取信息方式 有回显的注入 联合查询注入、基于报错…

Redis存储⑥Redis五大数据类型之 Zset

目录 1. Zset 有序集合 1.1 Zset 有序集合常见命令 zadd zcard zcount zrange zrevrange zrangebyscore&#xff08;弃用&#xff09; zpopmax bzpopmax zpopmin bzpopmin zrank zrevrank zscore zrem zremrangebyrank zremrangebyscore zincrby 1.2 Zset有…

青少年编程与数学 02-009 Django 5 Web 编程 06课题、模型定义

青少年编程与数学 02-009 Django 5 Web 编程 06课题、模型定义 一、模型二、定义模型1. 导入模型类2. 定义模型类3. 定义字段4. 添加元数据&#xff08;可选&#xff09;5. 定义模型方法&#xff08;可选&#xff09;6. 迁移模型 三、模型字段字符字段数字字段日期和时间字段布…

《探秘AI绿色计算:降低人工智能硬件能耗的热点技术》

在人工智能飞速发展的当下&#xff0c;其硬件能耗问题愈发凸显。据国际能源署预测&#xff0c;人工智能的能源消耗可能大幅增长。因此&#xff0c;降低人工智能硬件能耗&#xff0c;实现绿色计算&#xff0c;已成为行业关键课题。以下是一些正在崭露头角的热点技术。 新型硬件…

Linux之kernel(1)系统基础理论(2)

Linux之Kernel(1)系统基础理论(2) Author: Once Day Date: 2025年2月10日 一位热衷于Linux学习和开发的菜鸟&#xff0c;试图谱写一场冒险之旅&#xff0c;也许终点只是一场白日梦… 漫漫长路&#xff0c;有人对你微笑过嘛… 全系列文章可参考专栏: Linux内核知识_Once-Day的…

爬虫抓取过程的详细步骤

1. 目标网站分析 在开始编写爬虫之前&#xff0c;首先需要对目标网站进行详细的分析。这一步是整个爬虫开发过程中非常重要的环节&#xff0c;因为它直接决定了爬虫的效率和成功率。 确定目标数据&#xff1a;明确你想要抓取的数据&#xff0c;例如商品名称、价格、描述、图片…