问题说明:

假设数学表达式中允许包含两种括号:圆括号“()”和方括号“[]”,嵌套顺序任意。

正确的嵌套模式:( [ ] ( ) )、[ ( [ ] [ ] ) ]

正确的表达式例:(a+b)[c*(d-e)]

错误的嵌套模式:[ ( ] )、( ( ) ]

比如,在处理表达式(A)时

(A)  4+(2+8)*[5/(9-7)]

有以下步骤:

(1)检测到第一个括号“(”;

(2)检测到第二个括号“)”,说明子表达式 “4+(2+8)” 已完成匹配;

(3)检测到第三个括号“[”;

(4)检测到第四个括号“(”,与(3)中的括号不匹配,但由于同是左括号,可以继续匹配;

(5)检测到第五个括号“)”,由括号的作用可知,后来的括号比先来的括号优先级高,因此与(4)中括号匹配;

(6)检测到第六个括号“]”,由于原来优先级更高的括号已完成,因此与(3)中括号匹配。至此所有括号匹配完成。

解决方案:

可以看出,匹配成功的条件很简单:每一个检测到的括号与已检测到的优先级最高的括号都匹配。

匹配失败的条件:

(1)检测到与已检测到的优先级最高的括号不匹配的括号;

(2)扫描完整个表达式,还是有已检测到的括号没有完成匹配;

由于栈具有“先进后出”的特点,能很好地表现优先级这个性质,因此可以用栈来存储已经检测到的括号。

以(A)为例:

有以下步骤:

(1)检测到第一个括号“(”,进栈;

(2)检测到第二个括号“)”,进栈。子表达式 “4+(2+8)” 完成匹配,匹配的括号都出栈;

(3)检测到第三个括号“[”,进栈;

(4)检测到第四个括号“(”,进栈。与(3)中的括号不匹配,但由于同是左括号,可以继续匹配;

(5)检测到第五个括号“)”,进栈。由括号的作用可知,后来的括号比先来的括号优先级高,因此与(4)中括号匹配,匹配的括号都出栈;

(6)检测到第六个括号“]”,进栈。由于原来优先级更高的括号已完成,因此与(3)中括号匹配。匹配的括号都出栈,至此所有括号匹配完成。

需要注意的是,第一个括号进栈时,没有比较对象,因此需要特别处理。

判断函数 judge () 如下:

 1 Status judge(Stack *S)
2 {
3 //进行表达式的输入和判断
4 SElemType *p;
5 char n;
6
7 scanf("%c",&n); //第一个括号的检测
8 while(n!='(' && n!='[') //忽略数字等其他符号,直到输入括号
9 {
10 if(n==')' || n==']' || n=='#') return FALSE; //若为')'或']',则表明不匹配。'#'用于结束输入
11 scanf("%c",&n);
12 }
13
14 if(n=='(' || n=='[') Push(S,n); //检测到左括号,进栈
15 scanf("%c",&n);
16 while(n!='#') //'#'用于结束输入
17 {
18 if(n=='(' || n==')' || n=='[' || n==']')
19 {
20 p=S->top;
21 Push(S,n);
22 if(*(p-1)=='(') //与前一个括号比较
23 {
24 if(*p==')')
25 {
26 printf("%c\n",Pop(S));
27 printf("%c\n",Pop(S));
28 }
29 else if(*p==']') return FALSE;
30 }
31 else if(*(p-1)=='[')
32 {
33 if(*p==']')
34 {
35 printf("%c\n",Pop(S));
36 printf("%c\n",Pop(S));
37 }
38 else if(*p==')') return FALSE;
39 }
40 }
41 scanf("%c",&n);
42 }
43 if(S->top==S->base) return TRUE; //栈内没有元素时,说明匹配
44 else return FALSE;
45 }

源代码如下:

#include <stdio.h>
#include <stdlib.h>
#define INIT_SIZE 10
#define INCREMENT 5 #define OVERFLOW -2
#define FALSE 0
#define OK 1
#define TRUE 1
#define ERROR 0 typedef char SElemType;
typedef int Status;
typedef struct stack{
SElemType *base;
SElemType *top;
int stacksize;
}Stack; Status InitStack(Stack *S)
{
//初始化栈
S->base=(SElemType *)malloc(INIT_SIZE*sizeof(SElemType));
if(!S->base) exit(OVERFLOW);
S->top=S->base;
S->stacksize=INIT_SIZE;
return OK;
} Status Push(Stack *S,char e)
{
//入栈
if(S->top-S->base>=S->stacksize){ //栈满,重新分配内存
S->base=(SElemType *)realloc(S->base,(INIT_SIZE+INCREMENT)*sizeof(SElemType));
if(!S->base) exit(OVERFLOW);
S->top=S->base+S->stacksize;
S->stacksize+=INCREMENT;
}
*S->top++=e;
return OK;
} char Pop(Stack *S)
{
//出栈
char e; if(S->top==S->base) return ERROR; //栈空,出错
e=*(--S->top);
return e;
} Status judge(Stack *S)
{
//进行表达式的输入和判断
SElemType *p;
char n; scanf("%c",&n); //第一个括号的检测
while(n!='(' && n!='[') //忽略数字等其他符号,直到输入括号
{
if(n==')' || n==']' || n=='#') return FALSE; //若为')'或']',则表明不匹配。'#'用于结束输入
scanf("%c",&n);
} if(n=='(' || n=='[') Push(S,n); //检测到左括号,进栈
scanf("%c",&n);
while(n!='#') //'#'用于结束输入
{
if(n=='(' || n==')' || n=='[' || n==']')
{
p=S->top;
Push(S,n);
if(*(p-1)=='(') //与前一个括号比较
{
if(*p==')')
{
printf("%c\n",Pop(S));
printf("%c\n",Pop(S));
}
else if(*p==']') return FALSE;
}
else if(*(p-1)=='[')
{
if(*p==']')
{
printf("%c\n",Pop(S));
printf("%c\n",Pop(S));
}
else if(*p==')') return FALSE;
}
}
scanf("%c",&n);
}
if(S->top==S->base) return TRUE; //栈内没有元素时,说明匹配
else return FALSE;
} int main()
{
Stack Sta;
Stack *S=&Sta; printf("INITIALIZING...\n");
if(InitStack(S)) printf("DONE!\n"); printf("enter an expression(stop by '#'):\n");
if(judge(S)==1) printf("It's True!");
else printf("It's False\n"); return 0;
}

source

栈的应用:表达式括号匹配检测(C)的更多相关文章

  1. luoguP1739 表达式括号匹配 x

    P1739 表达式括号匹配 题目描述 假设一个表达式有英文字母(小写).运算符(+,—,*,/)和左右小(圆)括号构成,以“@”作为表达式的结束符.请编写一个程序检查表达式中的左右圆括号是否匹配,若匹 ...

  2. C++括号匹配检测(用栈)

    输入一串括号,包括圆括号和方括号,()[],判断是否匹配,即([]())或[([][])]为匹配的正确的格式,[(])或([())为不匹配的格式. #include<iostream> # ...

  3. java:数据结构(二)栈的应用(括号匹配)

    一.什么是括号匹配: 括号匹配就是利用计算机辨别表达式里面的括号是否书写成功 例如: {()((a)) }这就是一个正确 (()()   这就是一个错误的 二.括号匹配的算法: 众所周知,括号分为花括 ...

  4. 洛谷 P1739 表达式括号匹配

    题目链接https://www.luogu.org/problemnew/show/P1739 题目描述 假设一个表达式有英文字母(小写).运算符(+,—,*,/)和左右小(圆)括号构成,以“@”作为 ...

  5. P1739_表达式括号匹配(JAVA语言)

    思路:刚开始想用stack,遇到'('就push,遇到')'就pop,后来发现其实我们只需要用到栈里'('的个数,所以我们用一个变量统计'('的个数就好啦~ 题目描述 假设一个表达式有英文字母(小写) ...

  6. C++学习(三十一)(C语言部分)之 栈和队列(括号匹配示例)

    括号匹配测试代码笔记如下: #include<stdio.h> #include<string.h> #include <stdlib.h> #define SIZ ...

  7. python栈--字符串反转,括号匹配

    栈的实现: # 定义一个栈类 class Stack(): # 栈的初始化 def __init__(self): self.items = [] # 判断栈是否为空,为空返回True def isE ...

  8. java版括号匹配检测

    做一个空栈,读入字符直到结尾.如果读入一个封闭符号,空栈时报错;非空时弹出栈尾字符,如果不匹配则报错.否则读入为开放字符,压入栈中.最后如果栈空,返回true. 其中用到MyStack类,详情请见 p ...

  9. (栈 注意格式)P1739 表达式括号匹配 洛谷

    题目描述 假设一个表达式有英文字母(小写).运算符(+,—,*,/)和左右小(圆)括号构成,以“@”作为表达式的结束符.请编写一个程序检查表达式中的左右圆括号是否匹配,若匹配,则返回“YES”:否则返 ...

随机推荐

  1. 这个厉害了,阿里P7大佬都在看的SpringCloud 总结,帮你梳理全部知识点!

    微服务 微服务架构是一种以一些微服务来替代开发单个大而全应用的方法,每一个小服务运行在自己的进程里,并以轻量级的机制来通信, 通常是 HTTP RESTful API.微服务强调小快灵, 任何一个相对 ...

  2. Macos系统上怎么自动下载任务

    相对于Windows系统来说,好用的Mac下载工具就显得比较少了.Folx作为Mac下载工具中的佼佼者,其自动化下载功能受到很多Mac系统用户的欢迎. 随着高清影视的发展,很多影视资源体动辄就是1-2 ...

  3. FL Studio中的Fruity slicer采样器功能介绍

    本章节采用图文结合的方式来给大家介绍电音编曲软件FL Studio中的Fruity Slicer采样器的功能,感兴趣的朋友可一起来交流哦. Fruity slicer(水果切片器)插件是FL Stud ...

  4. guitar pro系列教程(十四):Guitar Pro教程之创建新乐谱后的设置

    前面的章节我们有对Guitar Pro的单个功能作介绍,对于初学作曲,且又是吉他初学者的朋友们来说,学完这些功能介绍,自己还不能融会贯通起来,创建了一个新的乐谱后,但是看起来还不是很满意,今天我们就创 ...

  5. C语言常用的一些转换工具函数!

    1.字符串转十六进制 代码实现: 2.十六进制转字符串 代码实现: 或者 效果:十六进制:0x13 0xAA 0x02转为字符串:"13AAA2" 3.字符串转十进制 代码实现: ...

  6. 编程语言输出“ Hello World ”,你真的都会了吗?

    Hello World 中文意思是『你好,世界』.因为<The C Programming Language>中使用它做为第一个演示程序,非常著名,所以后来的程序员在学习编程或进行设备调试 ...

  7. 2014_07_11_VGA基础及封装

    -- VR1201 Color Filter LCOS Microdisplays QVGA (320*240 Pixel) Color Filter LCOS Microdisplays -- VR ...

  8. 2019 ACM/ICPC North America Qualifier G.Research Productivity Index(概率期望dp)

    https://open.kattis.com/problems/researchproductivityindex 这道题是考场上没写出来的一道题,今年看看感觉简单到不像话,当时自己对于dp没有什么 ...

  9. Zabbix监控使用进阶

    1. Zabbix基于SNMP监控 1.1 zabbix-web所能指定的监控方式 ssh/telnet agent:master/agent SNMP:Simple Network Manageme ...

  10. PyQt+moviepy音视频剪辑实战1:多个音视频合成顺序播放或同屏播放的视频文件实现详解

    专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt+moviepy音视频剪辑实战 专栏:PyQt入门学习 老猿Python博文目录 老猿学5G博文目录 一. ...