c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。

2024-03-29 06:08

本文主要是介绍c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

      最近在学习数据结构与算法,学到栈这里,就基于栈实现了一个简答四则表达式算值的程序。平时我们写的那
种表达式就是中缀,而计算机处理中缀是不占优势的一般都是将中缀转成后缀再计算值,在这里我也利用这个思路,
将中缀表达式分为以下两部:
     A.中缀表达式转成后缀表达式,主要有以下几步:
 
a.初始化一个运算符栈  b.从左到右读取一个字符  c.如果是操作数就直接输出到后缀表达式  d.如果是左括号就压入操作符栈  e.如果是右括号,则把与之匹配的左括号之间的运算符全部输出  f.如果是运算符就分情况执行以下操作  1).如果栈为空就直接压入  2).如果栈中有运算符,切当前运算符比栈顶运算符优先级高就压入,否则就弹出栈顶运算符  ,并一直重复1)、2)直到栈顶运算符优先级低于待处理运算符或栈为空  3).重复b直到读取完毕 B.后缀表达式求值,这个就相对简单一些了: a.初始化一个操作数栈 b.依次读取后缀表达式,遇到运算数就压进运算数栈,遇到操作符就将栈顶两个数pop出来求值 然后push进栈。一直循环执行b知道后缀表达式读取完毕,最后操作数栈顶那个数就是表达式。 

下面就直接贴代码了,代码注释基本能够帮助快速读懂程序。

#include <iostream>
#include <stack>
using namespace std;int getPriority(char c);    //输入运算符返回优先级
bool isLeft(char c);        //判断是否是左括号
bool isRight(char c);       //判断是否是右括号
bool isOperator(char c);    //判断是否是操作符(+-*/)
bool isNumber(char c);      //判断是否是数字
double calculate(string postfix);   //传入后缀表达式,计算值
string convertToPostfix(const string& expression); //将中缀表达式转化为后缀表达式int main() {stack<char> operatorStack;string expression,postfix;cout << "请输入要计算的表达式:"<<endl;getline(cin,expression);postfix = convertToPostfix(expression);//输出结果cout << "后缀表达式为(数字之间以#号分割)->"<<postfix << endl;if (expression.at(expression.length()-1) == '=')cout <<expression<<calculate(postfix)<<endl;elsecout <<expression<<"="<<calculate(postfix)<<endl;
}string convertToPostfix(const string& expression){stack<char> operatorStack;string postfix;//依次读取输入的字符并按字符类型分别处理for(int current = 0; current < expression.length(); current++){char c = expression.at(current);if(!isNumber(c) && !isOperator(c) && c !='(' && c != ')'){cout << "请输入正确表达式!" << endl;exit(0);}if(c != ' ' && c != '='){//如果是左括号就压栈if(isLeft(c)) {operatorStack.push(c);} else if (isRight(c)){     //是右括号则将和左括号之间的pop出去拼接到后缀表达式while(!operatorStack.empty() && operatorStack.top() != '('){postfix.append(1,operatorStack.top());operatorStack.pop();}operatorStack.pop();        //最后将左括号pop} else if(isOperator(c)){   //如果是操作符if(current+1 < expression.length()){if (isOperator(expression.at(current+1))) {cout << "请输入正确的表达式";exit(0);}}if(current-1 >= 0){if(isOperator(expression.at(current-1))){cout << "请输入正确的表达式";exit(0);}}if(operatorStack.empty()){  //如果操作符栈为空就直接压栈operatorStack.push(c);} else{                     //否则就将栈顶优先级低的依次出栈,直到栈为空或者,栈顶运算符优先级低于当前运算符while(getPriority(c) <= getPriority(operatorStack.top())){postfix.append(1,operatorStack.top());operatorStack.pop();if(operatorStack.empty())break;}operatorStack.push(c);}} else {    //最后如果是数字的话就直接接到后缀表达式postfix.append(1,c);            //同时为了方便后面计算后缀表达式值的时候明确多位数的起止,特在每个整数后面加一个分隔符$if(current+1 < expression.length()){if (!isNumber(expression.at(current+1))){postfix.append(1,'#');//数字分隔符}}}}}//中缀表达式读取完毕后将运算符栈中剩下的运算符拼接到后缀表达式中while(!operatorStack.empty()){if(operatorStack.top() == '(' || operatorStack.top() == ')'){cout << "左右括号不匹配,表达式错误!";exit(0);}postfix.append(1,operatorStack.top());operatorStack.pop();}return postfix;
}
//返回运算符优先级
int getPriority(char c){if(c == '*' || c == '/')return 2;else if(c == '+' || c == '-')return 1;elsereturn  0;
}bool isRight(char c){if(c == ')')return true;elsereturn false;
}bool isLeft(char c){if(c == '(')return true;elsereturn false;
}bool isOperator(char c){if(c == '*' || c == '/' || c == '+' ||c == '-')return true;elsereturn false;
}double calculate(string postfix){stack<double> result;char c;for (int i = 0; i < postfix.length();i++) {c = postfix.at(i);if (isOperator(c)) {double a = result.top();result.pop();double b = result.top();result.pop();double temp;switch (c) {case '+' :temp = b + a;break;case '-' :temp = b - a;break;case '/' :if(a == 0){cout << "除零错误!";exit(0);}temp = b / a;break;case '*' :temp = b * a;default:break;}result.push(temp);} else if(isNumber(c)){double num = c - '0';while(isNumber(postfix.at(i+1))){num = num*10+ (postfix.at(i+1) - '0');i++;}result.push(num);}}return result.top();
}bool isNumber(char c){if(c>='0'&& c<='9')return true;elsereturn false;
}

当然代码中加入了一些判别表达式错误的片段,比如左右括号不匹配,或是除数为零,以及为了支持多位数,特意在每个
数结束处加一个#号方便后缀表达式计算值时识别,相关代码段如下:

            postfix.append(1,c); //同时为了方便后面计算后缀表达式值的时候明确多位数的起止,特在每个整数后面加一个分隔符#                if(current+1 < expression.length()){if (!isNumber(expression.at(current+1))){postfix.append(1,'#');//数字分隔符}}

 

 

 

 

 

 

 

 

#

 

这篇关于c++利用栈简单实现四则中缀表达式转后缀表达式,并算值。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/857805

相关文章

Java中使用Java Mail实现邮件服务功能示例

《Java中使用JavaMail实现邮件服务功能示例》:本文主要介绍Java中使用JavaMail实现邮件服务功能的相关资料,文章还提供了一个发送邮件的示例代码,包括创建参数类、邮件类和执行结... 目录前言一、历史背景二编程、pom依赖三、API说明(一)Session (会话)(二)Message编程客

Java中List转Map的几种具体实现方式和特点

《Java中List转Map的几种具体实现方式和特点》:本文主要介绍几种常用的List转Map的方式,包括使用for循环遍历、Java8StreamAPI、ApacheCommonsCollect... 目录前言1、使用for循环遍历:2、Java8 Stream API:3、Apache Commons

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

C#提取PDF表单数据的实现流程

《C#提取PDF表单数据的实现流程》PDF表单是一种常见的数据收集工具,广泛应用于调查问卷、业务合同等场景,凭借出色的跨平台兼容性和标准化特点,PDF表单在各行各业中得到了广泛应用,本文将探讨如何使用... 目录引言使用工具C# 提取多个PDF表单域的数据C# 提取特定PDF表单域的数据引言PDF表单是一

使用Python实现高效的端口扫描器

《使用Python实现高效的端口扫描器》在网络安全领域,端口扫描是一项基本而重要的技能,通过端口扫描,可以发现目标主机上开放的服务和端口,这对于安全评估、渗透测试等有着不可忽视的作用,本文将介绍如何使... 目录1. 端口扫描的基本原理2. 使用python实现端口扫描2.1 安装必要的库2.2 编写端口扫

PyCharm接入DeepSeek实现AI编程的操作流程

《PyCharm接入DeepSeek实现AI编程的操作流程》DeepSeek是一家专注于人工智能技术研发的公司,致力于开发高性能、低成本的AI模型,接下来,我们把DeepSeek接入到PyCharm中... 目录引言效果演示创建API key在PyCharm中下载Continue插件配置Continue引言

MySQL分表自动化创建的实现方案

《MySQL分表自动化创建的实现方案》在数据库应用场景中,随着数据量的不断增长,单表存储数据可能会面临性能瓶颈,例如查询、插入、更新等操作的效率会逐渐降低,分表是一种有效的优化策略,它将数据分散存储在... 目录一、项目目的二、实现过程(一)mysql 事件调度器结合存储过程方式1. 开启事件调度器2. 创

使用Python实现操作mongodb详解

《使用Python实现操作mongodb详解》这篇文章主要为大家详细介绍了使用Python实现操作mongodb的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、示例二、常用指令三、遇到的问题一、示例from pymongo import MongoClientf

SQL Server使用SELECT INTO实现表备份的代码示例

《SQLServer使用SELECTINTO实现表备份的代码示例》在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误,在SQLServer中,可以使用SELECTINT... 在数据库管理过程中,有时我们需要对表进行备份,以防数据丢失或修改错误。在 SQL Server 中,可以使用 SE

基于Go语言实现一个压测工具

《基于Go语言实现一个压测工具》这篇文章主要为大家详细介绍了基于Go语言实现一个简单的压测工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录整体架构通用数据处理模块Http请求响应数据处理Curl参数解析处理客户端模块Http客户端处理Grpc客户端处理Websocket客户端