用C实现一个计算器(带小数点和括号以及运算优先级)

/*************************************************************************
功能:利用栈结构做一个简易的计算器
描述:括号为最高优先级,其次乘除为次优先级,加减法的优先级最低
作者:566
**************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX 100

typedef struct stack1                      //数据栈
{
	float num[MAX];
	int top;
}O_NUM;

typedef struct stack2                      //符号栈
{
	char symbol[MAX];
	int top;
}O_SY;

/**********************************************************************
描述:初始化数据栈和符号栈
参数:O_NUM *stacknum,O_SY *stacksymbol
返回值:void
***********************************************************************/
void init_O_NUM(O_NUM *stacknum)                //初始化数据栈
{
	stacknum->top = -1;
}

void init_O_SY(O_SY *stacksymbol)               //初始化符号栈
{
	stacksymbol->top = -1;
}
/*************************************************************************
描述:将操作数推进操作数栈中
参数:O_NUM *stacknum,float num
返回值:int
**************************************************************************/
int PUSH_Stacknum(O_NUM *stacknum,float num)
{
	if(stacknum->top == MAX-1)
	{
		printf("已经超过输入长度,输入失败!\n");
		return -1;
	}
	stacknum->top++;
	stacknum->num[stacknum->top] = num;
	return 0;
}
/***************************************************************************
描述:将符号推进符号栈中
参数:O_SY *stacksymbol, char ch
返回值:int
***************************************************************************/
int PUSH_Stacksymbol(O_SY *stacksymbol,char ch)
{
	if(stacksymbol->top == MAX-1)
	{
		printf("已经超过输入长度,输入失败!\n");
		return -1;
	}
	stacksymbol->top++;
	stacksymbol->symbol[stacksymbol->top] = ch;
	return 0;
}
/***************************************************************************
描述:从操作数栈中取出数据
参数:O_NUM *stacknum
返回值:float
***************************************************************************/
float POP_Stacknum(O_NUM *stacknum)
{
	float data = 0;
	data = stacknum->num[stacknum->top];
	stacknum->top--;
	return data;
}
/**************************************************************************
描述:从符号栈中取出数据
参数:O_SY *stacksymbol
返回值:char
**************************************************************************/
char POP_Stacksymbol(O_SY *stacksymbol)
{
	char ch;
	ch = stacksymbol->symbol[stacksymbol->top];
	stacksymbol->top--;
	return ch;
}
/*************************************************************************
描述:定义函数judge_symbol判断运算符数否输入正确
参数:char ch
返回值:1(正确),0(错误)
*************************************************************************/
int judge_symbol(char ch)
{
	if(ch == '+'||ch == '-'||ch == '*'|| ch == '/'||ch == '('||ch == ')'||ch == '='||ch == '\n')
	{
		return 1;
	}
	else
	{
		return 0;
	}
}
/*************************************************************************
描述:判断运算符与栈顶运算符的优先级
参数:char t1,char t2      t1为栈顶指针,t2为与栈顶比较的指针
返回值:char
**************************************************************************/
char op_campare(char t1,char t2)
{
	char ch;
	switch(t2)
	{
		case '+':
		case '-':
		{
			ch = (t1 == '('||t1 == '=')?'<':'>';
			break;
		}
		case '*':
		case '/':
		{
			ch = (t1 == '*'||t1 == '/' ||t1 == ')')?'>':'<';
			break;
		}
		case '(':
		{
		 	if(t1 == ')')
		 	{
				printf("input error\n");
				return -1;
		 	}
			else
			{
				ch = '<';
			}
			break;
		}
		case ')':
		{
			switch(t1)
			{
				case '(':
				{
					ch = '=';
					break;
				}
				case '=':
				{
					printf("error input\n");
					exit(0);
				}
				default:ch = '>';
			}
			break;
		}
		case '=':
		{
			switch(t1)
			{
				case '=':
				{
					ch = '=';
					break;
				}
				case '(':
				{
					printf("error input!\n");
					return -1;
				}
				default:ch = '>';
			}
		}
	}
	return ch;
}
/**********************************************************************
描述:简单的计算功能
参数:float num1,char op,float num2
返回值:float
***********************************************************************/
float calculate(float num1,char op,char num2)
{
	float result;
	switch(op)
	{
		case '+':
		{
			result = num1 + num2;
			break;
		}
		case '-':
		{
			result = num1 - num2;
			break;
		}
		case '*':
		{
			result = num1 * num2;
			break;
		}
		case '/':
		{
			if(num2 == 0)
			{
				printf("error input!\n");
				return -1;
			}
			else
			{
				result = num1/num2;
			}
		}
	}
	return result;
}
/**************************************************************************
描述用于获取用户输入,并且进行计算
参数:void 
返回值:float result
***************************************************************************/
float MainCalc()
{
	O_NUM stacknum;                               //操作数
	O_SY stacksymbol;                             //运算符
	float num1,num2,result,num;
	char ch,sign;
	char *str = NULL; 
	int count = 0;

	init_O_NUM(&stacknum);                        //初始化
	init_O_SY(&stacksymbol);

	//将'='计算在运算符栈中
	PUSH_Stacksymbol(&stacksymbol,'=');
	ch = getchar();                              //从键盘读取字符
	while((ch != '=')||stacksymbol.symbol[stacksymbol.top] != '=')
	{
		if(judge_symbol(ch) == 0)
		{
			str = (char *)malloc(sizeof(char)*10); 
			do
			{
				*str = ch;
				str++;
				count++;
				ch = getchar();
			}while(judge_symbol(ch) == 0);
			*str = '\0';
			str = str - count;
			num = atof(str);                     //将字符串转换为浮点数
			PUSH_Stacknum(&stacknum,num);        //将数推进操作数栈中
			str = NULL;
			count = 0;
		}
		else
		{
			switch(op_campare(stacksymbol.symbol[stacksymbol.top],ch))
			{
				case '<':
				{
					PUSH_Stacksymbol(&stacksymbol,ch);
					ch = getchar();
					break;
				}
				case '=':
				{
					sign = POP_Stacksymbol(&stacksymbol);
					ch = getchar();
					break;
				}
				case '>':
				{
					sign = POP_Stacksymbol(&stacksymbol);
					num2 = POP_Stacknum(&stacknum);
					num1 = POP_Stacknum(&stacknum);
					result = calculate(num1,sign,num2);
					PUSH_Stacknum(&stacknum,result);
					break;
				}
			}
		}
	}
	result = stacknum.num[stacknum.top];
	return result;
}

int main()
{
	float result;

	result = MainCalc();
	printf("%g\n",result);
	return 0;
}

  • 9
    点赞
  • 86
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值