利用栈实现对后缀表达式(逆波兰表达式)的求解(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

运行效果截图如下:
在这里插入图片描述

  • 21
    点赞
  • 124
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
为了保证算法的正确性,我们首先要将中缀表达式转化成后缀表达式,也就是逆波兰表达式实现逆波兰表达式的常用方法是使用实现。 下面是代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> #define MAX_EXPR_LEN 100 // 表达式最大长度 #define MAX_STACK_SIZE 50 // 最大长度 // 结构体 typedef struct Stack { int top; int data[MAX_STACK_SIZE]; } Stack; // 初始化 void init(Stack *s) { s->top = -1; } // 入 void push(Stack *s, int data) { s->top++; s->data[s->top] = data; } // 出 int pop(Stack *s) { int data = s->data[s->top]; s->top--; return data; } // 判断是否为空 int isEmpty(Stack *s) { return (s->top == -1); } // 逆波兰表达式求解 int evaluate(char *expr) { Stack s; init(&s); int i, len = strlen(expr); int op1, op2; for (i = 0; i < len; i++) { if (isdigit(expr[i])) { push(&s, expr[i] - '0'); } else if (expr[i] == '+' || expr[i] == '-' || expr[i] == '*' || expr[i] == '/') { op2 = pop(&s); op1 = pop(&s); switch (expr[i]) { case '+': push(&s, op1 + op2); break; case '-': push(&s, op1 - op2); break; case '*': push(&s, op1 * op2); break; case '/': push(&s, op1 / op2); break; default: break; } } } return pop(&s); } int main() { char expr[MAX_EXPR_LEN]; printf("请输入表达式:"); scanf("%s", expr); printf("结果为:%d\n", evaluate(expr)); return 0; } ``` 使用方法: 1. 编译代码:`gcc filename.c -o filename`,filename 为你给代码起的名字 2. 运行代码:`./filename`,根据提示输入表达式 3. 程序将输出表达式的计算结果 注意: 1. 表达式长度不得超过100 2. 表达式中只能包含数字,加减乘除四种运算符 3. 表达式必须符合逆波兰表达式的格式要求 以上就是用 c 的和逆波兰算法实现达式求解代码实现

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值