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语言.其中多 ...
随机推荐
- 如何构思你的iOS App应用
转自:http://mobile.51cto.com/hot-311134.htm 在twitter上,听到不少人问,我也有App 在App Store上面卖,为什么我的App卖得这么少啊?于是我就上 ...
- flink batch wordcount
1.POJO方式 public class WordCountPojo { public static class Word{ private String word; private int fre ...
- MySQL数据库(一)-- 数据库介绍、MySQL安装、基础SQL语句
一.数据库介绍 1.什么是数据库 数据库即存储数据的仓库 2.为什么要用数据库 (1)用文件存储是和硬盘打交道,是IO操作,所以有效率问题 (2)管理不方便 (3)一个程序不太可能仅运行在同一台电脑上 ...
- 适用于Centos6/7,vsftp自动安装脚本
#!/bin/bash #vsftp install . /etc/rc.d/init.d/functions NUM=`rpm -q centos-release | awk -F '-' '{pr ...
- Java JMS——消息服务
转载请注明原文地址: https://www.cnblogs.com/ygj0930/p/10921569.html 一:什么是Java消息服务—— 消息通信接口规范 Java消息服务指的:两个应用程 ...
- Java字符串——String深入
转载请注明原文地址:https://www.cnblogs.com/ygj0930/p/10840495.html 一:字符串的不可变性 1.可变 与 不可变 辨析 Java中的对象按照创建后,对象的 ...
- maven仓库报错 sqljdbc4、ojdbc6、tomcat-jdbc-8.5.14
报错:Cannot resolve com.microsoft.sqlserver:sqljdbc4:4.0 和 Missing artifact com.microsoft.sqlserver: ...
- Linux操作系统安全-证书的申请原理
Linux操作系统安全-证书的申请原理 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.APR的中间人攻击 如下图所示,如果在client和server端有一个中间人攻击就比较麻 ...
- Nginx 核心配置-可优化配置参数
Nginx 核心配置-可优化配置参数 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.nginx的官网查看指令帮助信息方法 1>.打开nginx的官网(https://ng ...
- JavaScript/JQuery自执行函数
JavaScript中任何库与框架设计的第一个要点就是解决命名空间与变量污染的问题.jQuery就是利用了JavaScript函数作用域的特性,采用自执行函数包裹了自身的方法来解决这个问题.从jQue ...