C语言利用栈实现对后缀表达式的求解

所属分类: 软件编程 / C 语言 阅读数: 70
收藏 0 赞 0 分享

本文实例为大家分享了C语言实现对后缀表达式(逆波兰表达式)的求解代码,供大家参考,具体内容如下

逆波兰表达式:

逆波兰表达式又叫后缀表达式。它是由相应的语法树的后序遍历的结果得到的。
例:5 - 8*(6 + 7) + 9 / 4:

其中缀表达式为:5 - 8 * 6 + 7 + 9 / 4

其语法树如下:

因此根据语法树可以得出他后序遍历(后缀表达式)为:
5 8 6 7 + * - 9 4 / +

这样就实现了中缀表达式到后缀表达式的转换。
同样的也可以得出他的前序遍历(前缀表达式也称波兰表达式):
 + - 5 * 8 + 6 7 / 9 4

逆波兰表达式计算实现原理:
1.首先当遇到运算操作数时将其进行push操作;

2.当遇到操作符是将此时的栈pop两次,先取出的栈顶为右操作数;

3.执行此方法到整个数组遍历完。

实现算法如下:

void CalFunction(SqStack *S,char str[])
{/*实现浮点型数据后缀表达式的加减乘除*/
 Elemtype number,e,d;
 char arr[MAXBUFFER];
 int i=0,j=0;
 
 InitStack(S);
 
 while(str[i]!='\0')
 {
 while(isdigit(str[i])||str[i]=='.') //过滤数字
 {
 arr[j++]=str[i++];
 arr[j]='\0';
 
 if( j >= MAXBUFFER )
 {
 printf("输入单个数据过大!\n");
 return ;
 }
 if(str[i]==' ')
 {
 number=atof(arr); //利用atof函数将数字字符串转化为double型数据
 PushStack(S,number); //将转换的数进行压栈
 j=0;   //这里不要忘记将j重新初始化进行下个数据的转化
 break;
 }
 }
 /*如果遇到操作运算符则,弹出两个数据进行运算,然后将得出的结果重新入栈*/
 switch(str[i])
 {
 case '+':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d+e);
 break;
 case '-':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d-e);
 break;
 case '*':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d*e);
 break;
 case '/':
 PopStack(S,&e);
 PopStack(S,&d);
 if(e == 0)
 {
 printf("输入出错,分母为零!\n");
 return ;
 }
 PushStack(S,d/e);
 break;
 }
 i++; //继续遍历直到遍历字符串结束
 }
 
 PopStack(S,&e);
 printf("计算结果为:%lf",e); 
}

完整代码如下:

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<ctype.h>

#define INITSIZE 20
#define INCREMENT 10
#define MAXBUFFER 10
#define LEN sizeof(Elemtype)

/*栈的动态分配顺序存储结构*/
typedef double Elemtype;
typedef struct{
 Elemtype *base;
 Elemtype *top;
 int StackSize; 
}SqStack;

void InitStack(SqStack *S)
{
 S->base=(Elemtype*)malloc(LEN*INITSIZE);
 assert(S->base != NULL);
 S->top=S->base;
 S->StackSize=INITSIZE;
}

void PushStack(SqStack *S,Elemtype e)
{
 if(S->top - S->base >= S->StackSize)
 {
 S->base=(Elemtype*)realloc(S->base,(S->StackSize+INCREMENT)*LEN);
 assert(S->base !=NULL);
 S->top=S->base+S->StackSize;
 S->StackSize+=INCREMENT;
 }
 *S->top =e;
 S->top++;
}

void PopStack(SqStack *S,Elemtype *e)
{
 *e=*--S->top;
}

void CalFunction(SqStack *S,char str[])
{
 Elemtype number,e,d;
 char arr[MAXBUFFER];
 int i=0,j=0;
 
 InitStack(S);
 
 while(str[i]!='\0')
 {
 while(isdigit(str[i])||str[i]=='.') //过滤数字
 {
 arr[j++]=str[i++];
 arr[j]='\0';
 
 if( j >= MAXBUFFER )
 {
 printf("输入单个数据过大!\n");
 return ;
 }
 if(str[i]==' ')
 {
 number=atof(arr); //利用atof函数将数字字符转化为double型数据
 PushStack(S,number); //将转换的数进行压栈
 j=0;
 break;
 }
 }
 
 switch(str[i])
 {
 case '+':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d+e);
 break;
 case '-':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d-e);
 break;
 case '*':
 PopStack(S,&e);
 PopStack(S,&d);
 PushStack(S,d*e);
 break;
 case '/':
 PopStack(S,&e);
 PopStack(S,&d);
 if(e == 0)
 {
 printf("输入出错,分母为零!\n");
 return ;
 }
 PushStack(S,d/e);
 break;
 }
 i++; 
 }
 
 PopStack(S,&e);
 printf("计算结果为:%lf",e); 
}

int main()
{
 char str[100];
 SqStack S;
 printf("请按逆波兰表达式输入数据,每个数据之间用空格隔开:");
 gets(str);
 CalFunction(&S,str);
 return 0;
}


// 检测用例 5 - (6 + 7) * 8 + 9 / 4

// 输入:5 8 6 7 + * - 9 4 / + # 

// 输出: - 96.750000

运行效果截图如下:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

C++无法重载点符号、::、sizeof等的原因

这篇文章主要介绍了C++无法重载点符号、::、sizeof等的原因的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

实例讲解C语言编程中的结构体对齐

这篇文章主要介绍了C语言编程中的结构体对齐,值得注意的是一些结构体对齐的例子在不同编译器下结果可能会不同,需要的朋友可以参考下
收藏 0 赞 0 分享

详解C语言的结构体中成员变量偏移问题

这篇文章主要介绍了C语言的结构体中成员变量偏移问题,以讲解如何编写宏来对成员变量进行修改为主,需要的朋友可以参考下
收藏 0 赞 0 分享

详解C语言结构体中的函数指针

这篇文章主要介绍了详解C语言结构体中的函数指针,文中对函数指针的基本概念也有讲解,需要的朋友可以参考下
收藏 0 赞 0 分享

C++编程中的函数指针初步解析

这篇文章主要介绍了C++编程中的函数指针初步解析,函数指针在C语言和C++学习中都是非常重要的知识,需要的朋友可以参考下
收藏 0 赞 0 分享

实例解析C++中类的成员函数指针

这篇文章主要介绍了C++中类的成员函数指针,例子中以讨论用函数指针调用类的成员函数为主,需要的朋友可以参考下
收藏 0 赞 0 分享

C语言中的函数指针基础学习教程

这篇文章主要介绍了C语言中的函数指针基础学习教程,包括函数指针作为参数来传递等重要知识,需要的朋友可以参考下
收藏 0 赞 0 分享

深入解析C语言中函数指针的定义与使用

这篇文章主要介绍了C语言中函数指针的定义与使用,是C语言入门学习中的基础知识,需要的朋友可以参考下
收藏 0 赞 0 分享

详解C语言编程中的函数指针以及函数回调

这篇文章主要介绍了C语言编程中的函数指针以及函数回调,函数回调实际上就是让函数指针作函数参数、调用时传入函数地址,需要的朋友可以参考下
收藏 0 赞 0 分享

C语言中的函数指针学习笔记

这篇文章主要介绍了C语言中的函数指针的一些学习知识点记录,文中作者整理了一些比较interesting的函数指针用法,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多