数据结构-栈和队列的应用(验证括号的正确性,表达式求值,层次遍历)

本文主要是介绍数据结构-栈和队列的应用(验证括号的正确性,表达式求值,层次遍历),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

栈和队列的应用

  栈的应用

  验证括号的正确性

  题目很简单就是输入一串字符,判断字符中的括号是否合法。直接上代码:

#include <iostream>
#include <string.h>
using namespace std;typedef char ElemType;
#define MAXSIZE 100typedef struct Stack
{ElemType data[MAXSIZE];int top;
}Stack;void InitStack(Stack& S) {S.top = -1;
}bool StackEmpty(Stack S) {if (S.top == -1) {return true;}return false;
}bool Push(Stack& S, ElemType x) {if (S.top == MAXSIZE - 1) {return false;}S.top++;S.data[S.top] = x;return true;
}
bool Pop(Stack& S, ElemType& x) {if (S.top == -1) {return false;}x = S.data[S.top];S.top--;return true;
}ElemType GetTop(Stack S) {return S.data[S.top];
}int main() {Stack S;InitStack(S);string str;cin >> str;//flag标志状态 true为括号匹配,false为不匹配bool flag = true;for (int i = 0; i < str.size(); i++) {//元素若为{,(,[则入栈if ((str[i] == '{') || (str[i] == '[') || (str[i] == '(')) {Push(S, str[i]);}//元素若为},),]则出栈 赋值给rightif ((str[i] == '}') || (str[i] == ']') || (str[i] == ')')) {if ((str[i] == '}' && GetTop(S) == '{') || (str[i] == ']' && GetTop(S) == '[') || (str[i] == ')' && GetTop(S) == '(')) {char top = Pop(S, top);continue;}else {Push(S, str[i]);}}}if (S.top != -1) {    //当栈不为空时flag = false;}if (flag == false) {cout << "括号不匹配!" << endl;}else cout << "括号匹配!" << endl;system("pause");return 0;
}

  实验结果:

image-20210122155026920

image-20210122155116917

  表达式求值

  要求一个表达式的结果,例如 9 +(3 - 1) * 3 + 1 ,我们可以直接算出来,但是计算机不会,计算机一般将这种表达式转换成后缀表达式,运算符在操作数后面,并且运算也是根据优先级排列好的,没有括号,利于计算机的计算,如上述表达式转换为后缀表示式为:

9 3 1 - 3 * 10 + ,但是我们现在要实现的是中缀表达式的求值。计算思路:

  • 使用两个栈,stack0用于存储操作数,stack1用于存储操作符
  • 从左往右扫描,遇到操作数入栈stack0
  • 遇到操作符时,如果优先级低于或等于栈顶操作符优先级,则从stack0弹出两个元素进行计算,并压入stack0,继续与栈顶操作符的比较优先级
  • 如果遇到操作符高于栈顶操作符优先级,则直接入栈stack1
  • 遇到左括号,直接入栈stack1,遇到右括号,则直接出栈并计算,直到遇到左括号
//算符优先法
#include<iostream>
#include<cstring>
#include<stack>
using namespace std;  
const int maxn=110; 
char priority[7][7]={ {'>','>','<','<','<','>','>'},  {'>','>','<','<','<','>','>'},  {'>','>','>','>','<','>','>'},  {'>','>','>','>','<','>','>'},  {'<','<','<','<','<','=','0'},   // 此行"("=")"表示左右括号相遇,括号内运算已完成 {'>','>','>','>','0','>','>'},  {'<','<','<','<','<','0','='}    // "=" 表示整个表达式求值完毕 };                               //  "0"表示不可能出现这种情况 ( 语法错误 ) //Precede 用于判断运算符栈栈顶运算符 a1 与读入运算符 a2 之间的优先关系函数 
char Procede(char a,char b){   // 建立 pre[][] 到 运算符间的映射关系 int i,j;  switch(a){  case'+':i=0;break;  case'-':i=1;break;  case'*':i=2;break;  case'/':i=3;break;  case'(':i=4;break;  case')':i=5;break;  case'#':i=6;break;   // # 是表达式的结束符 }  switch(b){  case'+':j=0;break;  case'-':j=1;break;  case'*':j=2;break;  case'/':j=3;break;  case'(':j=4;break;  case')':j=5;break;  case'#':j=6;break;  }  return priority[i][j];  
}int Operate(int m,int n,char x){  if(x=='+')  return m+n;  if(x=='-')  return n-m;  if(x=='*')  return m*n;  if(x=='/')  return n/m;  
}  
// EvaluuateExpression-reduced
int main(){stack <int> OPND;  // Operand stackstack <char> OPTR;  // Operator stackOPTR.push('#');//char ss[2]="#";//尾部有\0 char s[maxn];cin>>s;strcat(s,ss);// 运算式尾部加 "#"--结束运算符 char c=s[0];int k=1;while(c!='#'||OPTR.top()!='#'){  //表达式未读完或者运算未完 int y=0;  if(c>='0'&&c<='9'){    while(c>='0'&&c<='9'){  // 读入连续的数字 y=y*10+(c-'0');  c=s[k++];  }  OPND.push(y);  // 把读进的数字入数字栈 }else{switch(Procede(OPTR.top(),c))  {  case'<':  //栈顶元素优先权低 OPTR.push(c);  c=s[k++];  break;  case'=':  OPTR.pop();  // 脱括号 c=s[k++];  // 读入下一个字符 break;  case'>':  //退栈并将运算结果入栈 char x=OPTR.top();OPTR.pop();  int m=OPND.top();OPND.pop();  int n=OPND.top();OPND.pop();  OPND.push(Operate(m,n,x));  break;    } }}cout<<OPND.top();return 0;
}

  实验结果:

image-20210122155312932

  队列的应用

  层次遍历

  利用队列存储每一层的结点,再存储到数组中,很容易理解。下面是套路:

vector<vector<int>> levelOrder(TreeNode* root) {queue<TreeNode*> q;if (root != nullptr)   q.push(root);vector<vector<int>> res;while (!q.empty()) {int sz = q.size();vector<int> temp;for (int i = 0; i < sz; i++) {TreeNode* cur = q.front();q.pop();temp.push_back(cur->val);if (cur->left != nullptr)q.push(cur->left);if (cur->right != nullptr)q.push(cur->right);}res.push_back(temp);}return res;
}

  队列在计算机系统中的应用

  一、解决主机与外部设备之间速度不匹配的问题。

  二、解决由多用户引起的资源竞争问题。

如果觉得本文对你有帮助的话,不妨关注作者一波,小小的关注其实对我很重要。更多高质量内容与资料请访问:数据结构简单学,个人主页:修心的小屋
如果喜欢的话,不妨关注一波,谢谢啦。
在这里插入图片描述

这篇关于数据结构-栈和队列的应用(验证括号的正确性,表达式求值,层次遍历)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/qq_44285551/article/details/112987551
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/652908

相关文章

从基础到高阶详解Python多态实战应用指南

《从基础到高阶详解Python多态实战应用指南》这篇文章主要从基础到高阶为大家详细介绍Python中多态的相关应用与技巧,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、多态的本质:python的“鸭子类型”哲学二、多态的三大实战场景场景1:数据处理管道——统一处理不同数据格式

Java Stream 的 Collectors.toMap高级应用与最佳实践

《JavaStream的Collectors.toMap高级应用与最佳实践》文章讲解JavaStreamAPI中Collectors.toMap的使用,涵盖基础语法、键冲突处理、自定义Map... 目录一、基础用法回顾二、处理键冲突三、自定义 Map 实现类型四、处理 null 值五、复杂值类型转换六、处理

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Python标准库之数据压缩和存档的应用详解

《Python标准库之数据压缩和存档的应用详解》在数据处理与存储领域,压缩和存档是提升效率的关键技术,Python标准库提供了一套完整的工具链,下面小编就来和大家简单介绍一下吧... 目录一、核心模块架构与设计哲学二、关键模块深度解析1.tarfile:专业级归档工具2.zipfile:跨平台归档首选3.

使用IDEA部署Docker应用指南分享

《使用IDEA部署Docker应用指南分享》本文介绍了使用IDEA部署Docker应用的四步流程:创建Dockerfile、配置IDEADocker连接、设置运行调试环境、构建运行镜像,并强调需准备本... 目录一、创建 dockerfile 配置文件二、配置 IDEA 的 Docker 连接三、配置 Do

深入浅出SpringBoot WebSocket构建实时应用全面指南

《深入浅出SpringBootWebSocket构建实时应用全面指南》WebSocket是一种在单个TCP连接上进行全双工通信的协议,这篇文章主要为大家详细介绍了SpringBoot如何集成WebS... 目录前言为什么需要 WebSocketWebSocket 是什么Spring Boot 如何简化 We

Java Stream流之GroupBy的用法及应用场景

《JavaStream流之GroupBy的用法及应用场景》本教程将详细介绍如何在Java中使用Stream流的groupby方法,包括基本用法和一些常见的实际应用场景,感兴趣的朋友一起看看吧... 目录Java Stream流之GroupBy的用法1. 前言2. 基础概念什么是 GroupBy?Stream

python中列表应用和扩展性实用详解

《python中列表应用和扩展性实用详解》文章介绍了Python列表的核心特性:有序数据集合,用[]定义,元素类型可不同,支持迭代、循环、切片,可执行增删改查、排序、推导式及嵌套操作,是常用的数据处理... 目录1、列表定义2、格式3、列表是可迭代对象4、列表的常见操作总结1、列表定义是处理一组有序项目的

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C#中的Converter的具体应用

《C#中的Converter的具体应用》C#中的Converter提供了一种灵活的类型转换机制,本文详细介绍了Converter的基本概念、使用场景,具有一定的参考价值,感兴趣的可以了解一下... 目录Converter的基本概念1. Converter委托2. 使用场景布尔型转换示例示例1:简单的字符串到