C语言之四则运算表达式求值(链栈)—支持浮点型数据,负数, 整型数据运算
运算符间的优先级关系:

链栈结构体定义:
数据域使用字符串长度为20的字符数组(故需要注意判断读取的字符串是运算符还是数值)
可支持浮点型数据,负数, 整型数据的运算
float EvaluateExpression() 函数实现步骤:
1)初始化OPTR栈和OPND栈,将表达式起始符 “#” 压入OPTR栈。
2)扫描表达式,读入第一个字符串str,如果表达式没有扫描完毕至 "#" 或压入OPTR的栈顶元素不为 "#" 时,则循环执行以下操作:
——>使用 str_to_float() 函数判断输入的字符串str是否是运算符
——>如果str不是运算符,则压入OPND栈,读取下一个字符串str
——>如果字符串str是运算符,使用 Precede() 函数获取OPTR栈顶元素的运算符和字符串str的运算符的优先级:
——>若是 ‘<’ ,则字符串str压入OPTR栈,读入下一个字符串str
——>若是 ‘>’ ,则弹出OPTR栈顶的运算符字符串,从OPND栈弹出两个数值字符串,使用 Operate() 函数对两个字符串进行运算,将得到的浮点型运算结果使用 float_to_str() 函数转换成字符串型数据,压入OPND栈
——>若是 ‘=’ ,则OPTR的栈顶元素是 "(" 且 str 是 ")" ,这时弹出OPTR栈顶的 "(" ,相当于括号匹配成功,然后读入下一字符串str
3)OPND栈顶元素记为表达式求值结果,输出运算结果。
实现代码(.cpp后缀文件)
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h> #define inf float(0x3f3f3f3f)
#define MAXSIZE 100 char priority[] = {'+', '-', '*', '/', '(', ')', '#'}; char priority_relationship[][] = {
{'>', '>', '<', '<', '<', '>', '>'},
{'>', '>', '<', '<', '<', '>', '>'},
{'>', '>', '>', '>', '<', '>', '>'},
{'>', '>', '>', '>', '<', '>', '>'},
{'<', '<', '<', '<', '<', '=', ' '},
{'>', '>', '>', '>', ' ', '>', '>'},
{'<', '<', '<', '<', '<', ' ', '='}
}; // 各个运算符之间的优先级关系 int get_index(char str[])
{// 获取相应运算符的索引
for(int i = ; i < ; i++)
{
if(str[] == priority[i]) return i;
}
printf("未找到匹配的字符\n");
} char Precede(char inside_data[], char input_data[])
{// 获取两个运算符之间的优先级关系
int inside_index = get_index(inside_data);
int input_index = get_index(input_data); return priority_relationship[inside_index][input_index];
} typedef struct StackNode
{
char data[MAXSIZE]; // 压入栈里面的数据都是字符型,在进行运行时,记得将字符型数字转换为浮点型数字
struct StackNode *next;
}StackNode, *LinkStack; void InitStack(LinkStack &S)
{// 构造一个空栈S,栈顶指针置空
S = NULL;
} void Push(LinkStack &S, char data[])
{// 在栈顶插入元素data
StackNode *p; p = (StackNode *)malloc(sizeof(StackNode)); // 生成新的结点
strcpy(p->data, data); // 将新结点的数据域置为data
p->next = S; // 将新结点插入栈顶
S = p; // 修改栈顶指针为p
} char *Pop(LinkStack &S)
{// 删除S的栈顶元素, 用data返回其值
char data[MAXSIZE];
if(S == NULL) printf("错误!!!\n栈为空, 无法执行删除命令...");
else
{
StackNode *p; strcpy(data, S->data); // 将栈顶元素赋给data
p = S; // 用p临时保存栈顶元素的空间,以备释放
S = S->next; //修改栈顶指针
free(p); // 释放原栈顶元素的空间
return data;
}
} char *GetTop(LinkStack &S)
{// 获取栈顶元素
if(S != NULL)
return S->data;
else
{
printf("错误!!!\n栈顶为空");
return "";
}
} float str_to_float(char *str)
{// 将字符串数据转换成浮点型数据
float num = ;
int state_1 = ; // 用于判断是否读取到小数点的状态值, 0代表还没有读取到
int state_2 = ; //用于判断是否读取到负号的状态值, 0代表还没有读取到
while(( *str != '\0' && *str >= '' && *str <= '') || *str == '.' || (*str == '-' && *(str + ) != '\0'))
{// 注意判断小数点和负号
if(*str == '.') state_1 = ; // 当读取到小数点的时候, 状态值state_1赋值为1
else if(*str == '-') state_2 = ; // 当读取到负号的时候, 状态值state_2赋值为1
else
{
if(state_1 == ) num = num * + (*str - '');
else
{
num += (*str - '') * pow(0.1, state_1);
state_1++;
}
}
str++;
}
if(*str != '\0') return inf;
else if(state_2 == )
{
return num * -;
}
else return num;
} char *float_to_str(float num)
{// 将浮点型数据装换成字符串数据,保留浮点型数据小数点后4位的值
char str[MAXSIZE];
sprintf(str, "%.4f", num); // 保留小数点后4位
return str;
} float Operate(char a[], char theta[], char b[])
{//执行运算
float a_num = str_to_float(a);
float b_num = str_to_float(b); if(theta[] == '+') return a_num + b_num;
else if(theta[] == '-') return a_num - b_num;
else if(theta[] == '*') return a_num * b_num;
else if(theta[] == '/') return a_num / b_num;
else printf("错误!!!\n无该运算符");
} void EvaluateExpression()
{
StackNode *OPND, *OPTR;
char str[MAXSIZE];
char theta[MAXSIZE];
char a[MAXSIZE];
char b[MAXSIZE]; InitStack(OPND); // 初始化栈 OPND
InitStack(OPTR); // 初始化栈 OPTR
Push(OPTR, "#"); // 将 "#" 压入栈OPTR printf("请输入算术表达式(支持负数,浮点型数据),每个值用空格隔开输入,并以#结束\n");
scanf("%s", str);
while(str[] != '#' || GetTop(OPTR)[] != '#')
{
if(str_to_float(str) != inf)
{ // 如果str不是运算符,则压入OPND栈,读取下一个字符串str
Push(OPND, str); // 将字符串str压入OPTR栈
scanf("%s", str); // 读入下一个字符串str
}
else
{ // 如果字符串str是运算符,使用 Precede() 函数获取OPTR栈顶元素的运算符和字符串str的运算符的优先级
switch (Precede(GetTop(OPTR), str)) // 使用 Precede() 函数获取相应优先级
{
case '<':
Push(OPTR, str); // 将字符串str压入OPTR栈
scanf("%s", str); // 读入下一个字符串str
break;
case '>':
strcpy(theta, Pop(OPTR)); // 弹出OPTR栈顶的运算符字符串并赋值给变量 theta
strcpy(b, Pop(OPND)); // 弹出OPND栈顶的数值字符串并赋值给变量 b
strcpy(a, Pop(OPND)); // 弹出OPND栈顶的数值字符串并赋值给变量 a
char temp_str[MAXSIZE];
strcpy(temp_str, float_to_str(Operate(a, theta, b))); // 根据相应的三个字符串进行运算,把结果赋给temp_str
Push(OPND, temp_str); // 将运算结果 temp_str 压入OPND栈
break;
case '=':
Pop(OPTR); // 弹出OPTR栈顶的运算符字符串
scanf("%s", str); // 读入下一个字符串str
break;
default:
printf("错误!!!\n该优先级不存在!!!");
}
}
}
printf("%s\n", GetTop(OPND));
} int main(void)
{
EvaluateExpression(); system("pause");
return ;
}
运行结果:

C语言之四则运算表达式求值(链栈)—支持浮点型数据,负数, 整型数据运算的更多相关文章
- 数据结构课程设计四则运算表达式求值(C语言版)
本系统为四则运算表达式求值系统,用于带小括号的一定范围内正负数的四则运算标准(中缀)表达式的求值.注意事项: 1.请保证输入的四则表达式的合法性.输入的中缀表达式中只能含有英文符号"+ ...
- C/C++ 语言中的表达式求值
在此,首先向裘老师致敬! 裘宗燕:C/C++ 语言中的表达式求值 经常可以在一些讨论组里看到下面的提问:“谁知道下面C语句给n赋什么值?” m = 1; n = m+++m++; 最近有位不相识的朋友 ...
- C/C++ 语言中的表达式求值(原文作者:裘宗燕)
经常可以在一些讨论组里看到下面的提问:“谁知道下面C语句给n赋什么值?”m = 1; n = m+++m++;最近有位不相识的朋友发email给我,问为什么在某个C++系统里,下面表达式打印出两个4, ...
- 表达式求值(栈方法/C++语言描述)(二)
上篇中完成了对表达式求值的整体过程,接下来看看如何处理不同类型的token. 对运算数的处理比较简单,它直接调用函数strtod(),将字符串中的运算数转换为浮点类型并将它压入运算数栈中: void ...
- 【NYOJ-35】表达式求值——简单栈练习
表达式求值 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 Dr.Kong设计的机器人卡多掌握了加减法运算以后,最近又学会了一些简单的函数求值,比如,它知道函数min ...
- Matrix Chain Multiplication(表达式求值用栈操作)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1082 Matrix Chain Multiplication Time Limit: 2000/100 ...
- 河南省acm第九届省赛--《表达式求值》--栈和后缀表达式的变形--手速题
表达式求值 时间限制:1000 ms | 内存限制:65535 KB 难度:3 描述 假设表达式定义为:1. 一个十进制的正整数 X 是一个表达式.2. 如果 X 和 Y 是 表达式,则 X+Y, ...
- 表达式求值(栈方法/C++语言描述)(一)
一个算数表达式(以下简称为表达式)由运算数.运算符.左括号和右括号组成,定义一个枚举类型TokenType表示为: typedef enum { BEGIN, NUMBER, OPERATOR, LE ...
- x86汇编反编译到c语言之——(1)表达式求值及赋值语句
一. 反编译一种可能的实现方式 我们的目的是将多种平台的汇编如x86,ARM,6502反编译为c语言,所以实现时先将多种汇编转化为 特定虚拟机汇编语言,然后只需要将虚拟机汇编语言反编译为c语言.其中多 ...
随机推荐
- Windows下VS Code打开黑屏解决办法(这样真的行)
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明.本文链接:https://blog.csdn.net/qq_42421611/article/d ...
- ADO.NET 根据实体类自动生成添加修改语句仅限Oracle使用
话不多说直接上代码,新手上路,高手路过勿喷,请多多指教. /// <summary> /// 等于号 /// </summary> ) + Convert.ToChar() + ...
- Qt Examples - Boxes (在Qt场景视图中结合OpenGL渲染)
QT自带例程Boxes使用QT Graphics View框架实现了2D图形和3D图形的混合渲染,综合性比较强,整合知识较多,值得学习. 可以使用鼠标通过以下方式控制演示中的元素: 按住鼠标左键的同时 ...
- iOS开发工具:Alcatraz、SVGKit、Lin以及Transformifier等
转自:http://www.cocoachina.com/applenews/devnews/2013/0606/6352.html Alcatraz:Xcode包管理器 Alcatraz是一个开源的 ...
- django framework插件使用1
安装 REST框架要求以下内容: Python(3.5.3.6.3.7) Django(1.11.2.0.2.1.2.2) pip install djangorestframework pip in ...
- rocketmq 两主两从异步集群搭建
1.安装JDK 需要先卸载系统默认的OPENJDK,安装 JDK1.8 64位的版本. 卸载open-jdk rpm -qa|grep java 查到open jdk的安装. 使用命令 rpm -e ...
- TCP 通信时序及状态变迁
TCP 通信时序及状态变迁 参考链接: https://www.cnblogs.com/boxker/p/11214886.html https://blog.csdn.net/miss_ruoche ...
- The 2019 Asia Nanchang First Round Online Programming Contest E. Magic Master
题目链接:https://nanti.jisuanke.com/t/41352 题目意思还是好理解的,看过的人不多,感觉是被通过量吓到了.其实就是个水题,反向模拟就好了, 用队列模拟,反向模拟,它要放 ...
- python 和 R 语言 中的 range() 函数
1.python 中的 range() 函数生成整数序列,常用于 for 循环的迭代. 示例: 2.R 语言中的 range() 函数返回一个数值向量中的最小值和最大中,常用于求极差. 示例: 按语: ...
- bootstrap的selectpicker的方法
方法 .selectpicker('val') 您可以通过调用val元素上的方法来设置所选值. 1 2 $('.selectpicker').selectpicker('val', 'Mustard' ...