第十五届蓝桥杯大赛 国赛 pb组F题【括号与字母】(15分) 栈的应用

server/2024/11/15 4:51:28/
  • 博客主页:誓则盟约
  • 系列专栏:IT竞赛 专栏
  • 关注博主,后期持续更新系列文章
  • 如果有错误感谢请大家批评指出,及时修改
  • 感谢大家点赞👍收藏⭐评论✍ 

试题F:括号与字母

【问题描述】

         给定一个仅包含小写字母和括号的字符串 S ,保证括号可以两两匹配。 给出 Q 组询问,每组询问给出一个小写字母 ci 和一个数 xi ,询问 S 中有 多少对匹配的括号之间有不少于 xi 个 ci 。

【输入格式】

        输入的第一行包含一个字符串 S 。 第二行包含一个整数 Q 。 接下来 Q 行,每行包含一个小写字母 ci 和一个整数 xi 表示一组询问,用 一个空格分隔。

【输出格式】

         输出 Q 行,每行包含一个整数,依次表示每个询问的答案。

【样例输入】

((a)()((b)((c))))

3

a 2

b 1

c 1

【样例输出】

0

3

4

【评测用例规模与约定】

        对于 40% 的评测用例,|S |, Q ≤ 5000 ; 对于 70% 的评测用例,|S | ≤ 100000 ; 对于所有评测用例,1 ≤ |S | ≤ 106 ,1 ≤ Q ≤ 100000 ,0 ≤ xi < 106 。其中 |S | 表示 S 的长度。


分析问题:

        仔细读题,保证给的s中括号都两两匹配,那么这道题就相当于是在考察入栈和出栈的问题了,这里我们需要定义一个符号栈和一个字母栈。

  • 符号栈:专门用来存储左括号,遇见左括号则入栈,遇见右括号则栈顶的左括号出栈。并且对加入的左括号所包含的字母个数做标记,记录出栈前的左括号和右括号之间有几个字母,最后可以通过字符串切割来找到这个括号内的字母。
  • 字母栈:遇见字母则入栈,不需要出栈。用于储存字母。


 

代码实现:

s=str(input()) # 输入s
s0="abcdefghijklmnopqrstuvwxyz" # 一会判断字母要用
q=int(input())  # 输入询问次数q
for i in range(q): # q次循环,每次询问都有一个输出值s1,b1=map(str,input().split())  # 输入要询问的字母和被包括的个数b=int(b1)  # 转次数为int型v=s.count(s1) if v<b:print(0)  # 此时总个数都小于要询问的个数b,一定没有符合题意的 返回0else:re=0 # 记录个数stick_1=[] # 符号栈stick_2=[] # 字母栈for j in s: # 遍历sif j=="(":  #遇见左括号则入栈stick_1.append([j,0]) # 后面的0 用于标记这个左括号与对应的右括号之间有几个字母elif j in s0:  # 如果是字母,则入字母栈stick_2.append(j)for w in range(len(stick_1)): stick_1[w][-1]+=1 # 对于所有的左括号对应的标记值,都加1,说明他们与对应的右括号之间多了一个字母else: # 否则则是右括号k_1,st=stick_1.pop() # 遇见右括号,则从符号栈出栈一个左括号if st==0: continue # 说明左括号右括号之间没有元素,直接跳ve=stick_2[-1:-st-1:-1] # 否则说明之间有元素,则找到这些元素if ve.count(s1)>=b:# 判断ve中要查询的字母个数是否合题意re+=1 # 标记的个数+1print(re) # 对于每次询问都返回 res

 

总结:

以下是对这段代码的详细解释:

  • s = str(input()):获取用户输入的字符串 s
  • s0 = "abcdefghijklmnopqrstuvwxyz":定义了所有小写字母的字符串,用于后续判断字母。
  • q = int(input()):获取询问的次数。
  • 然后进入 q 次循环:
    • s1, b1 = map(str, input().split()):分别获取要询问的字母和期望的包含个数,将 b1 转换为整数类型。
    • v = s.count(s1):计算字符串 s 中该字母出现的总次数。如果总次数小于期望个数,直接输出 0
    • 否则,进行复杂的处理:
      • re = 0 用于记录符合条件的个数。
      • stick_1 是符号栈,stick_2 是字母栈。
      • 遍历字符串 s
        • 遇到左括号,将其及初始标记值 0 入栈。
        • 遇到字母,入字母栈,并更新符号栈中每个左括号对应的标记值,表示它们之间多了一个字母。
        • 遇到右括号,弹出符号栈中的一个左括号和标记值。如果标记值为 0,则直接跳过;否则,找到左括号和右括号之间的元素,判断其中要查询的字母个数是否满足条件,如果满足则增加标记个数 re
      • 最后输出每次询问对应的 re

        总的来说,这段代码主要是通过栈的操作来处理字符串中括号内的子串,并判断其中特定字母的出现次数是否满足要求。

对于这道题的考点和反思如下:

考查内容

  1. 对字符串的处理和操作能力,包括字符的统计、遍历等。
  2. 栈这种数据结构的运用,通过栈来处理括号内的内容和计数。
  3. 逻辑思维和问题分析解决能力,需要仔细思考如何在复杂的条件下准确判断符合要求的情况。

 

学会的内容

  1. 更加深入地掌握了字符串处理的技巧和方法。
  2. 熟悉了栈的实际应用场景,以及如何通过栈来解决特定问题。
  3. 提升了面对复杂逻辑问题时设计算法和代码实现的能力。

反思

  1. 在处理复杂逻辑时,要更加仔细地设计算法和流程,避免遗漏特殊情况。
  2. 对于数据结构的运用要更加灵活,根据具体问题选择合适的数据结构来优化解决方案。
  3. 编写代码时要注意代码的可读性和可维护性,以便后续的理解和修改。同时要充分考虑代码的效率和性能。

        总之,这道题放在15分的位置,并不算是难题,主要还是考察对栈的应用熟练程度是否到位。 相关栈的篇章:栈的理解与应用

“甲之蜜糖,乙之砒霜。” ——《曼陀罗》


http://www.ppmy.cn/server/48548.html

相关文章

Qt6 播放音视频

一、概述 QT6相较于Qt5引入了许多新特性和改进&#xff0c;包括对音视频开发的增强支持。 QT6中的音视频支持 QT6提供了一套完整的音视频处理功能&#xff0c;这些功能被整合在QtAV项目中。QtAV是一个基于Qt的音视频处理框架&#xff0c;用于处理音视频播放、录制、编解码、处…

263.丑数

丑数 就是只包含质因数 2、3 和 5 的正整数。 给你一个整数 n &#xff0c;请你判断 n 是否为 丑数 。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 1&#xff1a; 输入&#xff1a;n 6 输出&#xff1a;true 解释&#xff1a;6 2 3 示例 …

Linux “ 软件管理 “

软件管理 widows 安装 方法一&#xff1a; 双击exe安装包&#xff0c;就可以安装。 用exe安装的软件会破记录到注册表中。 注册会记录安装位置&#xff0c;软件名称。 方法二&#xff1a; 用绿色方式进行安装。 不用写到注册表中&#xff0c;因此无法在开始菜单里面查看和卸…

【C++】编译

三、C编译 前面给大家演示了如何从写C代码到编译代码再到执行代码的全过程。这个过程中非常重要的编译环节&#xff0c;被我们一个按钮或者一个ctrlF7快捷键就给带过了。其实这个环节非常重要&#xff0c;如果你非常了解这个环节&#xff0c;你开发源代码就会更加自信和清醒&a…

一文了解如何安全有效的进行PB级别的大数据迁移

在这个信息量爆炸的时代&#xff0c;处理PB级别的数据转移已成为常态&#xff0c;但对企业而言&#xff0c;这仍然是一个充满挑战的任务。今天&#xff0c;我们来探讨一下这个话题&#xff0c;看看在进行PB级数据转移时&#xff0c;需要留意哪些事项&#xff0c;可能会遇到哪些…

使用 PNPM 从 0 搭建 monorepo,测试并发布

1 目标 通过 PNPM 创建一个 monorepo&#xff08;多个项目在一个代码仓库&#xff09;项目&#xff0c;形成一个通用的仓库模板。 这个仓库既可以用于公司存放和管理所有的项目&#xff0c;也可以用于将个人班余的所有积累整合其中。 2 环境要求 核心是 PNPM 和 Node.js&…

oracle dataguard 从库 MRP 进程的状态是 WAIT_FOR_GAP

因主库归档日志未备份直接删除后&#xff0c;从库不能更新&#xff0c;19c版本以上&#xff0c;之前未打补丁&#xff0c;使用 RECOVER STANDBY DATABASE FROM SERVICE PRM180;之后&#xff0c;在执行 alter database recover managed standby database using current logfil…

Qt信号槽与函数直接调用性能对比

1. 测试方法 定义一个类Recv&#xff0c;其中包含一个成员变量num和一个成员函数add()&#xff0c;add()实现num的递增。 另一个类Send通过信号槽或直接调用的方法调用Recv的add函数。 单独开一个线程Watcher&#xff0c;每秒计算num变量的增长数值&#xff0c;作为add函数被调…