简单表达式求值——算符优先法

前言

周五加班的时候,在九度oj上练习了一道简单表达式求值的题目,用到了“算符优先法”,这里简单的记录一下

题目

题目描述:
    读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值。
输入:
    测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符,整数和运算符之间用一个空格分隔。没有非法表达式。当一行中只有0时输入结束,相应的结果不要输出。
输出:
    对每个测试用例输出1行,即该表达式的值,精确到小数点后2位。
样例输入:
1 + 2
4 + 2 * 5 - 7 / 11
0
样例输出:
3.00
13.36


思路

对表达式 4 + 2 × 5  - 7 / 11进行计算,需要了解算法四则运算的规则,即:
  • 先乘除,后加减
  • 从左到右计算

实现算法

  1. 为了实现算符优先法,可以使用两个工作栈。一个称作OPTR,用以寄存运算符;另一个称为OPRD,用以寄存操作数或者运算结果。
  2. 初始化两个栈,并且置栈空。(假设表达式是没有语法错误的)
  3. 依次读入表达式中每个字符,若是操作数则进OPTD栈,若是运算符,则和OPTR栈的栈顶运算符比较优先权后作相应操作,直至整个表达式求值完毕(即表达式指针到了最后一个字符并且OPTR栈为空)


代码实现(c语言顺序栈)

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define stacksize 201

struct optr
{
	char operator[stacksize];
	int top;
};
struct optd
{
	double data[stacksize];
	int top;
};

void pushStackOptr(struct optr *, char );
void pushStackOptd(struct optd *, double );
char getStackOptrTop(struct optr *);
int compareOperator(char, char);
char popStackOptr(struct optr *);
double popStackOptd(struct optd *);
double calculateData(double, double, char);

int main()
{
	struct optr *optr;
	struct optd *optd;
	int i, len;
	double sum, x, data1, data2;
	char string[201], opt, opt1, opt2;

	while(gets(string) && string[0] != '0')
	{
		//初始化栈、表达式长度
		len = strlen(string);
		optr = (struct optr *)malloc(sizeof(struct optr));
		optd = (struct optd *)malloc(sizeof(struct optd));
		if(optr == NULL || optd == NULL)
		{
			exit(EXIT_FAILURE);
		}
		optr->top = optd->top = 0;

		//计算表达式的值,使用“算符优先法”
		for(i = 0; i < len; i ++)
		{
			if(string[i] != ' ')
			{
				if(string[i] >= '0' && string[i] <= '9')
				{
					x = string[i] - '0';
					while(string[i + 1] >= '0' && string[i + 1] <= '9')
					{
						x = 10 * x + (string[i + 1] - '0');
						i ++;
					}
					//操作数进栈
					pushStackOptd(optd, x);
				}else
				{
					//获取栈顶运算符
					opt = getStackOptrTop(optr);
					switch(compareOperator(opt, string[i]))
					{
						case 0:
							//栈顶优先级低
							pushStackOptr(optr, string[i]);
							break;
						case 1:
							//栈顶优先级高
							opt = popStackOptr(optr);
							data1 = popStackOptd(optd);
							data2 = popStackOptd(optd);
							sum = calculateData(data1, data2, opt);
							pushStackOptd(optd, sum);
							i --;
							break;	
					}
				}
			}
		}

		//判断算符栈是否为空
		while(optr -> top != 0)
		{
			opt = popStackOptr(optr);
			data1 = popStackOptd(optd);
			data2 = popStackOptd(optd);
			sum = calculateData(data1, data2, opt);
			pushStackOptd(optd, sum);
		}

		//输出计算结果
		printf("%.2lf\n",sum);
	}
	return 0;
}

void pushStackOptr(struct optr *s, char opt)
{
	//判断栈满
	if(s->top - 1 == stacksize)
	{
		exit(EXIT_FAILURE);
	}
	s->operator[s->top ++] = opt;
}

void pushStackOptd(struct optd *s, double x)
{
	//判断栈满
	if(s->top - 1 == stacksize)
	{
		exit(EXIT_FAILURE);
	}
	s->data[s->top ++] = x;
}

char getStackOptrTop(struct optr *s)
{
	//判断栈空
	if(s->top == 0)
	{
		return '#';
	}
	return s->operator[s->top - 1];
}

int compareOperator(char opt1, char opt2)
{
	//默认opt1优先级高,opt2优先级高返回0
	int flag = 1;

	if(((opt1 == '+' || opt1 == '-') && (opt2 == '*' || opt2 == '/')) || opt1 == '#')
	{
		flag = 0;
	}
	return flag;
}

char popStackOptr(struct optr *s)
{
	//判断栈空
	if(s->top == 0)
	{
		exit(EXIT_FAILURE);
	}
	return s->operator[-- s->top];
}

double popStackOptd(struct optd *s)
{
	//判断栈空
	if(s->top == 0)
	{
		exit(EXIT_FAILURE);
	}
	return s->data[-- s->top];
}

double calculateData(double data1, double data2, char operator)
{
	switch(operator)
	{
		case '+':
			return data2 + data1;
		case '-':
			return data2 - data1;
		case '*':
			return data2 * data1;
		case '/':
			if(data1 == 0)
			{
				exit(EXIT_FAILURE);
			}
			return data2 / data1;
	}
}

你可能感兴趣的:(简单表达式求值——算符优先法)