逆波兰表达式[栈 C 语言 实现]

逆波兰表达式

逆波兰表达式又叫做后缀表达式。在通常的表达式中,二元运算符总是置于与之相关的两个运算对象之间,这种表示法也称为中缀表示。波兰逻辑学家J.Lukasiewicz于1929年提出了另一种表示表达式的方法,按此方法,每一运算符都置于其运算对象之后,故称为后缀表示。

 

a+b ---> a,b,+

a+(b-c) ---> a,b,c,-,+

a+(b-c)*d ---> a,b,c,-,d,*,+

a+d*(b-c)--->a,d,b,c,-,*,+

1、将一个中序表达式转化成为逆波兰表达式

 

构造两个栈S1,S2,S1用来存放表达式,S2用来暂时存放运算符,最后完成后,该栈是清空的。

(1)如果遇到的是数字,直接进栈S1

(2)如果遇到的是左括号,进栈S2

(3)如果遇到的是右括号,将S2中的运算符全部出栈压入S1中,注意括号不压入

(4)如果遇到的运算符

 

         1.如果此时栈S2为空,则直接将运算符加入到栈S2中;

         2.如果此时栈S2不为空,当前运算符的优先级大于等于栈顶运算符的优先级,那么直接入栈S2;

         3.如果此时栈S2不为空,当前运算符的优先级小于栈顶运算符的优先级,则将栈顶运算符一直出栈压入到栈S1中,  直到栈为空或者遇到一个运算符的优先级小于等于当前遍历的运算符的优先级,然后将该运算符压入到栈S2中。    

(5)遍历完整个中序表达式之后,如果栈S2中仍然存在运算符,那么将这些运算符依次出栈压入到栈S1中,直到栈为空。

2、利用逆波兰表达式求值

 

维护一个结果栈S3,该栈最后存放的是表达式的值。从左至右的遍历栈S1

(1)如果遇到的是数字,直接将数字压入到S3中

(2)如果遇到的是单目运算符,取S3栈顶的一个元素进行运算之后,将结果压入到栈S3中

(3)如果遇到的是双目运算符,取S3栈顶的两个元素,首先出栈的在左,后出栈的在右进行双目运算符的计算,将结果压入到S3中

遍历完整个栈S1,最后S3中的值就是逆波兰表达式的值。

 

栈实现表达式计算【数据结构】

思路:

所包含的运算符有‘+’,‘-’,‘*’,‘/’,‘(’,‘)’。

(1)建立两个栈,一个用来存储操作数,另一个用来存储运算符, 开始时在运算符栈中先压入‘/0’,一个表达式的结束符。

(2)然后从左至右依次读取表达式中的各个符号(操作数或者运算符);

(3)如果读到的是操作数直接存入操作数栈;

(4)如果读到的是运算符,则作进一步判断:

若读到的是‘/0’结束符,而且此时运算符栈的栈顶元素也是‘/0’结束符,则运算结束,输出操作数栈中的元素即为最后结果。

若读到的是‘(’或者读到的运算符的优先级比目前的运算符栈中的栈顶元素的优先级高,则将运算符直接存入运算符栈,继续读表达式中的下一个符号,重复步骤(3)和(4);

若读到的是‘)’,而且此时运算符栈的栈顶元素是‘(’结束符,则将运算符栈中的栈顶元素退出来,继续读表达式中的下一个符号,重复步骤(3)和(4);

若读到的运算符的优先级等于或小于之前的运算符的优先级,则从操作数中退出2个,从运算符中退出一个进行运算,将运算结果存入操作数栈;再把之前读到的运算符与目前的运算符栈顶比较,重复步骤(4)(即现在不读下一个元素);

 

/*
输入:
9+(3-1)*3+10/2 输出
20注意:其实注释的地方可以用来调试,本人主要根据大话数据结构的思路写出来的如果有什么地方错了,多谢提出。
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <iostream>
using namespace std;
struct stack{char data[101];int top;
};struct stack2{int data1[101];int top;
}; struct stack tak;
struct stack2 tak2;bool jia(char s){if(s=='+' || s=='-')return true;else return false;
}
bool ch(char s){if(s=='*' || s=='/')return true;else return false;
}
int number(int y, int x, char s){if(s=='+')return x + y;if(s=='-')return x - y;if(s=='*')return x * y;if(s=='/' )return x / y; 
}int main(){char s1[20];scanf("%s",&s1);getchar();tak.top = 0;tak2.top = 0;//中缀表达式转化为后缀表达式 for(int i=0;i<strlen(s1);i++){if(s1[i]>='0' && s1[i]<='9'){int tep = s1[i]-'0';while(s1[i+1]>='0' && s1[i+1]<='9'){tep *= 10;i++;tep += s1[i]-'0'; }
//			printf("%d ",tep);tak2.top++;tak2.data1[tak2.top] = tep;}else{if(tak.top==0 || tak.data[tak.top]=='('){tak.top++;tak.data[tak.top] = s1[i];}else{char temp = tak.data[tak.top];if( jia(temp) && (ch(s1[i]) || jia(s1[i]) || s1[i]=='(')  ){tak.top++;tak.data[tak.top] = s1[i];	 }else if(ch(temp) && (ch(s1[i]) || s1[i]=='(')){tak.top++;tak.data[tak.top] = s1[i];}else if(s1[i]==')'){while(tak.top>0){if(tak.data[tak.top]=='('){tak.top--;break;}
//						printf("%c ",tak.data[tak.top]);int t1 = tak2.data1[tak2.top];tak2.top--;int t2 = tak2.data1[tak2.top];int t3 = number(t1,t2,tak.data[tak.top]);tak2.data1[tak2.top] = t3;tak.top--;}}else if(ch(temp) && jia(s1[i])){while(tak.top>0){if(tak.data[tak.top]=='('){break;}
//						printf("%c ",tak.data[tak.top]);int t1 = tak2.data1[tak2.top];tak2.top--;int t2 = tak2.data1[tak2.top];int t3 = number(t1,t2,tak.data[tak.top]);tak2.data1[tak2.top] = t3;tak.top--;}tak.top++;tak.data[tak.top] = s1[i];}}	} }while(tak.top>0){
//		printf("%c ",tk.data[tak.top]);int t1 = tak2.data1[tak2.top];tak2.top--;int t2 = tak2.data1[tak2.top];int t3 = number(t1,t2,tak.data[tak.top]);tak2.data1[tak2.top] = t3;
//		cout<<"  t3 = "<<t3<<endl;tak.top--;}printf("%d",tak2.data1[tak2.top]);return 0; 
}

 

转载于:https://www.cnblogs.com/0526yao/p/10372092.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/464081.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FireFox不支持cursor:hand

这个在IE/Firefox下都支持可以使用cursor:pointer;转载于:https://www.cnblogs.com/kaixin110/archive/2007/08/22/865287.html

Red Hat 6.0 Installation Steps

1) 注册和下载 - 需要一个公司email地址来接收RH的确认邮件 https://cn.redhat.com/products/enterprise-linux/server/download.html 下载免费红帽企业 Linux 30 天评估版 - 注册完毕转入下载页面 https://access.redhat.com/downloads/ -选择免费评估版本, 收到确认邮件, 找到…

若有所思

今天出来​逛了下&#xff0c;没有准备文章&#xff0c;刚好一个朋友有一段思考&#xff0c;给大家分享下。深圳动物园是一个值得去的地方​-----佚名海边的夏天&#xff0c;太阳公公当空照&#xff0c;照的大地滚烫&#xff0c;海水湛蓝。人们都躲到树荫下&#xff0c;享受舶风…

HDU 4383 To The Moon 解题报告

HDU 4383 To The Moon 题意翻译 已知一个长为\(n\)的序列\(a\)&#xff0c;你需要进行下面的四种操作。 C l r d 将区间\([l,r]\)中的数加上\(d\),同时时间加\(1\) Q l r 查询当前时间区间\([l,r]\)中所有数的和 H l r t 查询时间为\(t\)时区间\([l,r]\)中所有数的和 B t 将当前…

第三章 阴阳的工作机制(1)

一、道生一&#xff0c;一生二&#xff0c;二生三&#xff0c;三生万物1.易有太极&#xff0c;是生两仪上面&#xff0c;对阴阳已经讨论了很多&#xff0c;对这个问题已经有所了解。现在我们来讨论"阴阳的工作机制"&#xff0c;弄清楚这个以后对理解《伤寒论》的许多…

tcp窗口滑动以及拥塞控制

转自&#xff1a;http://blog.chinaunix.net/uid-26275986-id-4109679.html TCP协议作为一个可靠的面向流的传输协议&#xff0c;其可靠性和流量控制由滑动窗口协议保证&#xff0c;而拥塞控制则由控制窗口结合一系列的控制算法实现。 一、滑动窗口协议 关于这部分自己不…

jquery 操作日期、星期、元素的追加

主要实现日期的显示&#xff0c;获取年月日&#xff0c;时分秒、星期、判断闰年<script language"javascript" >$(document).ready(function(){function show(){var mydatenew Date();var str "" mydate.getFullYear() "年"; …

简单易懂的芯片科普漫画,帮你打开高深的新技术大门

大家好&#xff0c;我是写代码的篮球球痴&#xff0c;今天给大家推荐一本新书&#xff0c;是华为麒麟团队出版的&#xff0c;主要是让大家了解芯片的结构&#xff0c;这对很多人理解计算机是非常有帮助的。希望大家喜欢。喜欢书籍的同学在文章下面评论&#xff0c;我们会选出评…

UDT协议详细分析

UDT协议的主要特性有哪些&#xff1f; 基于UDP的应用层协议&#xff1a; 有基本网络知识的朋友都知道TCP和UDP的区别和使用场景&#xff0c;但是有没有一种协议能同时兼顾TCP协议的安全可靠和UDP协议的高效&#xff0c;那么UDT就是一种。 面向连接的协议&#xff1a;面向连接…

书摘---创业36条军规1:创业是怎么回事

这本书是电子版&#xff0c;是我这几年唯一购买的一本电子版本的图书。一来是想尝尝新鲜&#xff0c;二来是想看看书中有什么可以学习的东西。 本人一直在打工&#xff0c;所以不太了解创业的情况&#xff0c;站在巨人的肩膀上才能看的更远&#xff0c;看了此书不代表我就想创业…

在STM32上模拟Linux自动初始化过程

Linux中有很多编程思想可以学习&#xff0c;很多大佬把这些思想、机制运用到单片机的编程上&#xff0c;STM32 模拟Linux kernel自动初始化流程。通常我们写程序都是按照这个套路&#xff0c;一个函数一个函数按照顺序逻辑一个一个的执行下去。如果逻辑非常复杂&#xff0c;涉及…

日志配置(springboot、mybatis、Lombok)

Spring Boot在所有内部日志中使用Commons Logging&#xff0c;但是默认配置也提供了对常用日志的支持&#xff0c;如&#xff1a;Java Util Logging&#xff0c;Log4J, Log4J2和Logback。每种Logger都可以通过配置使用控制台或者文件输出日志内容 SLF4J——Simple Logging Faca…

先知

他这样说爱情&#xff1a;Love gives naught but itself and takes naught but from itself.Love possesses not nor would it be possessed.For love is sufficient unto love.转载于:https://www.cnblogs.com/belial/archive/2007/09/23/903433.html

TCP与UDP的选择--结合QQ来说明

TCP与UDP的选择如果比较UDP包和TCP包的结构&#xff0c;很明显UDP包不具备TCP包复杂的可靠性与控制机制。与TCP协议相同&#xff0c;UDP的源端口数和目的端口数也都支持一台 主机上的多个应用。一个16位的UDP包包含了一个字节长的头部和数据的长度&#xff0c;校验码域使其可以…

Xampp修改数据库默认用户root的密码

Xampp数据库默认用户root&#xff0c;密码为空&#xff0c;修改密码的位置&#xff1a;xampp/passwords.txt转载于:https://www.cnblogs.com/testlife/archive/2012/02/08/2342438.html

1.6元的蓝牙芯片,你能想得到?

前不久喵了个咪在群里发了一个1.6元的SOP8蓝牙芯片的信息&#xff0c;引起了大家的热烈讨论&#xff0c;我当然也按捺不住鸡冻的心情&#xff0c;立马下单买了几片。淘宝连接&#xff1a;KT6368Ahttps://item.taobao.com/item.htm?spma1z09.2.0.0.74a52e8dTpp1my&id630430…

pyplot交互地画多个plot

下面的代码&#xff0c;可以无阻碍地show 5个figure&#xff0c;相当于开启了ipython的interactive 模式 具体参见stackoverflow: in matplotlib, is there a way to pop up a figure asynchronously? 1 #!/usr/bin/python2 import pylab as plb3 import matplotlib.pyplot as…

又是一年中秋节,好想举杯邀明月

世事一场大梦&#xff0c;人生几度新凉。夜来风叶已鸣廊&#xff0c;看取眉间鬓上。 酒贱常愁客少&#xff0c;月明多被云妨。中秋谁与共孤光&#xff0c;把盏凄然北望。 简策写虚名&#xff0c; 蝼蚁侵枯骨。千古光阴一霎时&#xff0c;且进杯中物。 二十四年过去&#xff…

UDT内部代码分析

一. 报文发送 1.CSndQueue::worker中调用CChannel::sendto发送数据报文。 2.CSndQueue::sendto中调用CChannel::sendto发送其他报文, 种类较多主要有: 1)CUDT::connect中调用CSndQueue::sendto发送建立连接请求。 2) CUDT::sendCtrl中调用CSndQueue::sendto发送控制报文。 …

ADC采样效应及相关影响解剖

在前述文章&#xff0c;BUCK电路模拟补偿器的数字化过程 &#xff0c;我们讨论了模拟补偿器的数字化&#xff0c;事实上&#xff0c;数字化过程的第一个重要的环节就是ADC对反馈量的采样&#xff0c;本文就重点探讨一下由于ADC采样频率带来的一些问题&#xff0c;进而讨论一下相…