最新数学表达式计算(c语言实现)演示教学

合集下载

第2章_数据类型、运算符和表达式《C语言程序设计(第三版)》-电子教案

第2章_数据类型、运算符和表达式《C语言程序设计(第三版)》-电子教案

2.2 标识符、常量与变量

符号常量
【例2-1】输入圆的半径,计算周长和面积。 #define PI 3.14159 void main() { float r,c,s; scanf("%f",&r); c=2*PI*r; s=PI*r*r; printf("r=%f,c=%f,s=%f\n",r,c,s); }

以下是合法的整型常量:

2.2 标识符、常量与变量

实型常量

实型常量只能用十进制形式表示 表示形式:


小数形式。由数字序列和小数点组成,如 3.1415926、-0.15、.15、2.等都是合法的实型 常量。 指数形式。由十进制数加上阶码标志“e”或 “E”及阶码组成,如3.14e-4或3.14E-4表示 3.14×10-4。
2.2 标识符、常量与变量

2.2.3 变量


在程序运行过程中,其存储的值可以被改变的量称 为变量 变量必须通过标识符进行说明,称为变量名。 变量名和内存单元地址存在映射关系,程序可以通 过变量名寻址,从而访问其存储的数据。
2.2 标识符、常量与变量

变量的定义和说明

数据类型 变量名1[,变量名2,…,变量名n];
long型转换成float型时由原来可达10位整数变成只有7位有效数字精度丢失但由于数的范围扩大了数据类型从较低级提升到较高级随着竞争日益激烈酒店嘴中的肥肉被大肆抢夺各大酒店在这场竞争中几乎溃不成军
21世纪高等学校精品规划教材
第2章 数据类型、运算符 和表达式



掌握C语言的基本数据类型 掌握标识符、关键字、常量和变量的使用 掌握运算符、表达式以及数据类型之间的转换 等 掌握简单数据的输入输出

(完整版)数学表达式计算(c语言实现)

(完整版)数学表达式计算(c语言实现)

一、设计思想计算算术表达式可以用两种方法实现:1.中缀转后缀算法此算法分两步实现:先将算术表达式转换为后缀表达式,然后对后缀表达式进行计算.具体实现方法如下:(1)中缀转后缀需要建一个操作符栈op和一个字符数组exp,op栈存放操作符,字符数组用来存放转换以后的后缀表达式。

首先,得到用户输入的中缀表达式,将其存入str数组中。

对str数组逐个扫描,如果是数字或小数点,则直接存入exp数组中,当扫描完数值后,在后面加一个#作为分隔符。

如果是操作符,并且栈为空直接入栈,如果栈不为空,与栈顶操作符比较优先等级,若比栈顶优先级高,入栈;如果比栈顶优先级低或相等,出栈将其操作符存到exp数组中,直到栈顶元素优先等级低于扫描的操作符,则此操作符入栈;如果是左括号,直接入栈,如果是右括号,出栈存入exp数组,直到遇到左括号,左括号丢掉。

然后继续扫描下一个字符,直到遇到str中的结束符号\0,扫描结束。

结束后看op栈是否为空,若不为空,继续出栈存入exp数组中,直到栈为空.到此在exp数组最后加结束字符\0。

我们就得到了后缀表达式。

(2)后缀表达式计算此时需要一个数值栈od来存放数值。

对exp数组进行逐个扫描,当遇到数字或小数点时,截取数值子串将其转换成double类型的小数,存入od栈中。

当遇到操作符,从栈中取出两个数,进行计算后再放入栈中。

继续扫描,知道扫描结束,此时值栈中的数值就是计算的结果,取出返回计算结果。

2。

两个栈实现算法此算法需要两个栈,一个值栈od,一个操作符栈op。

将用户输入的数学表达式存入str数组中,对其数组进行逐个扫描。

当遇到数字或小数点,截取数值子串,将其转换成double类型的数值存入od栈中;当遇到左括号,直接入op栈;遇到右括号,op栈出栈,再从值栈od中取出两个数值,计算将其结果存入值栈中,一直进行此操作,直到操作符栈栈顶为左括号,将左括号丢掉。

如果遇到操作符,若op栈为空,直接入栈;若栈不为空,与栈顶元素比较优先等级,若比栈顶操作符优先等级高,直接入op栈,如果低于或等于栈顶优先等级,op栈出栈,再从值栈中取出两个数值,计算将其结果存入值栈中,一直进行此操作,直到栈顶优先等级低于扫描的操作符等级,将此操作符入op 栈。

C语言表达式和算术运算符ppt课件

C语言表达式和算术运算符ppt课件
如:17%-3=2 -19%4=-3 -15%-7=-1
5%1.5是非法的算术表达式
2)除法运算符“/”进行求商运算。对于不同类型的运算对象, 除法表达式计算结果的类型也会不同。
例:计算x/y 如果x,y为整型,小数部分舍去,没有四舍五入 如果x,y中有一个为实型量,则x,y都被化为double类型进
结合性: 同一优先级,自 左向右,为左结合性,反 之为右结合性。
初等运算符( [ ]、( )、 . 、-> )
↓ 单目运算符
↓ 算术算
符!)
↓ 赋值运算符
↓ 逗号运算符
.
4
3.5.2算术运算符和算术表达式
1、基本的算术运算符:
+ (加法运算符,或正值运算符。如:3+5、+3) - (减法运算符,或负值运算符。如:5-2、-3) * (乘法运算符。如:3*5) / (除法运算符。如:5/3) % (模运算符,或称求余运算符,%两侧均应为整型数据,
行计算,结果为double
.
8
算术运算符和算术表达式
运算实例: x,y为整型量:5/2结果为2,整型量
2/5结果为0,整型量 x,y其中一个为实型量 5.0/2或 5.0/2.0或 5.0/2.0结果为2.5,实型量
.
9
练习:
例 1/2 = 0 -5/2 = -2 -5/2.0 = -2.5
例 5%2 = 1 -5%2 = -1 5%-2 = 1 1%10 = 1 5%1 = 0 5.5%2 =
第3章 表达式和运算符(3)
.
1
3.5 运算符和表达式
3.5.1 运算符的种类、优先级和结合性 3.5.2 算术运算符和算术表达式 3.5.3 赋值运算符和赋值表达式 3.5.4 增量运算符和增量表达式 3.5.8 逗号运算符和逗号表达式

C语言运算符与表达式

C语言运算符与表达式

2021/2/15
7
如假设变量a的初值为4,设 c=(++a)*6;
a先加1得5参与运算(即5*6)得结果30。 若c=(a++)*6;
a先参与运算(即4*6)得结果是24。 在赋值表达式运算结束后a加1,因此a值最终也为5

关系运算符用来比较两个表达式值的大小,所
&(位与)、|(位或)、^(位异或) <<(左移)、>>(右移) 1个单目运算符: ~(取反)
整型常量或整型变量通过位运算符组成位运算 表达式。
例如int a=12,b=10则位运算如下: a 0000000000001100 (12) b 0000000000001010 (10)
a&b 0000000000001000 (8) a|b 0000000000001110 (14) a^b 0000000000000110 (6) a<<2 0000000000110000 (48) a>>2 0000000000000011 (3) ~b 1111111111110101 (-11) 当b是unsigned int类型时~b值为(65525)。
2021/2/15
陈孝则
1
• 按运算符所需操作数的多少可分:
• (1) 单目运算符—只需一个操作数。 • (2) 双目运算符—要求有两个操作数。 • (3) 三目运算符—要求有三个操作数。

• C语言中表达式可以是
• 1.一个常量 • 2.一个变量 • 3.由运算符连起来的常量变量
• 表达式可以含有运算符也可以不含有,根据使用运算符的 不同,表达式可分算术表达式、关系表达式、逻辑表达式、 赋值表达式、条件表达式和逗号表达式等。

(2024年)C语言程序设计教程完整全套教学课件pptx

(2024年)C语言程序设计教程完整全套教学课件pptx
示例
实现二维数组的动态内存分配、模拟命令行参数传 递等。
34
07
文件操作与数据处理
2024/3/26
35
文件概述及文件类型指针
文件概述
文件是存储在外部介质上的数据集合, 是程序设计中重要的数据存储和处理 方式。
文件类型指针
C语言中,文件类型指针用于指向文件 的指针变量,通过文件指针可以实现对 文件的读写操作。
2024/3/26
指针定义及本质
指针是一种特殊类型的变量,它存储的是另 一个变量的内存地址,而不是值本身。
指针基本操作
包括指针的赋值、取值、指针算术运算、指 针比较等。
31
指针作为函数参数传递
值传递与地址传递
通过值传递,函数接收参数的副本,对副本的修改不影响原始变量;通过地址传递,函 数直接操作原始变量的内存地址,可实现数据的修改。
介绍C语言编程的基本规范和风格,包括命名规则、 注释规则、缩进和空格的使用等。
2024/3/26
7
02
数据类型、运算符与表达式
2024/3/26
8
基本数据类型
01
整型(int)
02
浮点型(float、 double)
字符型(char)
03
04
布尔型(bool)
2024/3/26
9
变量与常量
变量
2024/3/26
3
C语言概述
C语言的历史与发展
介绍C语言的起源、发展历程以及在 计算机科学领域的重要地位。
C语言与高级语言的关系
探讨C语言与Java、Python等高级语 言之间的联系与区别,以及各自适用 的场景。
C语言的特点与优势
阐述C语言高效、灵活、可移植等特 点,以及在系统级编程、嵌入式开发 等领域的广泛应用。

C语言程序设计PPT课件第3章 简单的算术运算和表达式

C语言程序设计PPT课件第3章 简单的算术运算和表达式

算术表达式 (Arithmetic Expression)
Example:
( 9 – (–34 + 25) ) * 3 = ?1?2 ( 9 – ( 3 + 2 ) ) * 3 = 12
2019/9/9
18/51
赋值语句 (Assignment Statement)
三种赋值形式:
Simple——简单赋值 Multiple——多重赋值 Shorthand——简写的复合赋值
+=
num += 5;
num = num + 5;
-=
num -= 5;
num = num – 5;
*=
num *= 5;
num = num * 5;
/=
num /= 5;
num = num / 5;
%=
num %= 5;
num = num % 5;
简写的复合赋值(Shorthand Assignment)
2019/9/9
27/51
3.1.3增1和减1运算符
(Increment and Decrement)
n++,n--,++n,--n – ++让参与运算的变量加1,--让参与运算的变量减1 – 作为后缀(postfix)运算符时,先取n的值,然后加/减1
m = n++;
2019/9/9
m = n; n++;
14/51
算术表达式 (Arithmetic Expression)
优先级(Order of Precedence)
High: * / % Low: + -

C语言程序设计项目式教程完整版课件全书电子教案教材课件完整

C语言程序设计项目式教程完整版课件全书电子教案教材课件完整

例如,有一函数
,编写程序,其功能是对已知 x 求 y
程序说明 ① if 和 else 后面的语句可以是复合语句。 ②注意 if 与 else 的配对原则,else 总是与前面离它最近的没成对的 if 成对。
简单的C语言程序结构
说明4——语句
C语言中以“;”作为语句结束的标志。函数体就是由若干语句组成的,同时语句也出现在 函数之间,示例代码如下所示:
简单的C语言程序结构
学习一种编程语言,最佳途径就是多阅读代码段,多编写程序代码,接下来通过最基本的 C语言程序的基本构成学习基本格式和书写规范,代码示例如下所示:
单一if结构
选择结构中最基本的分支结构是 if 语句,按形式分 if 语句可以分为单分支、双分支和多 分支等,单一 if 语句定义形式如下所示
当上述中“表达式”值为“逻辑真”时,执行“语句”中内容。例如计算整型变量 x 的 绝对值,示例代码如下所示:
If-else结构
if 语句的第二种形式为 if-else 结构的双分支。其定义形式如下所示 : 当“表达式”值为“逻辑真”时,执行“语句 1”;当“表达式”值为“逻辑假”时,执 行“语句 2”。例如,判断整型变量 x 是 5,则输出“right”,否则输出“error”,示例代码如 下所示:
简单的C语言程序结构
说明2——预处理
include称为文件包含命令,其意义是把双引号""或尖括号<>内指定的文件包含到本程序来, 成为本程序的一部分。被包含的文件通常是由系统提供的,其扩展名为.h的头文件。C语言的 头文件中包括了各个标准库函数的函数原型。因此,凡是在程序中调用一个库函数时,都必 须包含该函数原型所在的头文,示例代码如下所示:
原型在头文件 “stdio.h”中。具体格式如下:

C#算术表达式求值(后缀法),看这一篇就够了

C#算术表达式求值(后缀法),看这一篇就够了

C#算术表达式求值(后缀法),看这⼀篇就够了⼀、种类介绍算术表达式有三种:前缀表达式、中缀表达式和后缀表达式。

⼀般⽤的是中缀,⽐如1+1,前后缀就是把操作符移到前⾯和后⾯,下⾯简单介绍⼀下这三种表达式。

1、前缀表⽰法前缀表⽰法⼜叫波兰表⽰法,他的操作符置于操作数的前⾯(例:+ 1 2),是波兰数学家扬·武卡谢维奇1920年代引⼊的,⽤于简化命题逻辑。

因为我们⼀般认为操作符是在操作数中间的,所以在⽇常⽣活中⽤的不多,但在计算机科学领域占有⼀席之地。

⼀般的表⽰法对计算机来说处理很⿇烦,每个符号都要考虑优先级,还有括号这种会打乱优先级的存在,将使计算机花费⼤量的资源进⾏解析。

⽽前缀表⽰法没有优先级的概念,他是按顺序处理的。

举个例⼦:9-2*3这个式⼦,计算机需要先分析优先级,先乘后减,找到2*3,再进⾏减操作;化成前缀表⽰法就是:- 9 * 2 3,计算机可以依次读取,操作符作⽤于后⼀个操作数,遇到减就是让9减去后⾯的数,⽽跟着9的是乘,也就是说让9减去乘的结果,这对计算机来说很简单,按顺序来就⾏了。

2、中缀表⽰法这也就是我们⼀般的表⽰法,他的操作符置于操作数的中间(例:1 + 2),前⾯也说过这种⽅法不容易被计算机解析,但他符合⼈们的普遍⽤法,许多编程语⾔也就⽤这种⽅法了。

在中缀表⽰法中括号是必须有的,要不然运算顺序会乱掉。

3、后缀表⽰法后缀表⽰法⼜叫逆波兰表⽰法,他的操作符置于操作数的后⾯(例:1 2 +),他和前缀表⽰法都对计算机⽐较友好,但他很容易⽤堆栈解析,所以在计算机中⽤的很多。

他的解释过程⼀般是:操作数⼊栈;遇到操作符时,操作数出栈,求值,将结果⼊栈;当⼀遍后,栈顶就是表达式的值。

因此逆波兰表达式的求值使⽤堆栈结构很容易实现,且能很快求值。

注意:逆波兰记法并不是简单的波兰表达式的反转。

因为对于不满⾜交换律的操作符,它的操作数写法仍然是常规顺序,如,波兰记法/ 6 3的逆波兰记法是6 3 /⽽不是3 6 /;数字的数位写法也是常规顺序。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

一、设计思想计算算术表达式可以用两种方法实现:1.中缀转后缀算法此算法分两步实现:先将算术表达式转换为后缀表达式,然后对后缀表达式进行计算。

具体实现方法如下:(1)中缀转后缀需要建一个操作符栈op和一个字符数组exp,op栈存放操作符,字符数组用来存放转换以后的后缀表达式。

首先,得到用户输入的中缀表达式,将其存入str数组中。

对str数组逐个扫描,如果是数字或小数点,则直接存入exp数组中,当扫描完数值后,在后面加一个#作为分隔符。

如果是操作符,并且栈为空直接入栈,如果栈不为空,与栈顶操作符比较优先等级,若比栈顶优先级高,入栈;如果比栈顶优先级低或相等,出栈将其操作符存到exp数组中,直到栈顶元素优先等级低于扫描的操作符,则此操作符入栈;如果是左括号,直接入栈,如果是右括号,出栈存入exp数组,直到遇到左括号,左括号丢掉。

然后继续扫描下一个字符,直到遇到str中的结束符号\0,扫描结束。

结束后看op栈是否为空,若不为空,继续出栈存入exp数组中,直到栈为空。

到此在exp数组最后加结束字符\0。

我们就得到了后缀表达式。

(2)后缀表达式计算此时需要一个数值栈od来存放数值。

对exp数组进行逐个扫描,当遇到数字或小数点时,截取数值子串将其转换成double类型的小数,存入od栈中。

当遇到操作符,从栈中取出两个数,进行计算后再放入栈中。

继续扫描,知道扫描结束,此时值栈中的数值就是计算的结果,取出返回计算结果。

2.两个栈实现算法此算法需要两个栈,一个值栈od,一个操作符栈op。

将用户输入的数学表达式存入str数组中,对其数组进行逐个扫描。

当遇到数字或小数点,截取数值子串,将其转换成double类型的数值存入od栈中;当遇到左括号,直接入op栈;遇到右括号,op栈出栈,再从值栈od中取出两个数值,计算将其结果存入值栈中,一直进行此操作,直到操作符栈栈顶为左括号,将左括号丢掉。

如果遇到操作符,若op栈为空,直接入栈;若栈不为空,与栈顶元素比较优先等级,若比栈顶操作符优先等级高,直接入op栈,如果低于或等于栈顶优先等级,op栈出栈,再从值栈中取出两个数值,计算将其结果存入值栈中,一直进行此操作,直到栈顶优先等级低于扫描的操作符等级,将此操作符入op栈。

继续扫描直到遇到str中的结束字符\0,扫描结束。

此时看操作符栈是否为空,若不为空,出栈,再从值栈中取出两个数值进行计算,将其结果存入值栈,一直进行此操作,直到操作符栈为空。

此时把值栈中的数值取出,即为所得的最终计算结果。

二、算法流程图第一种算法:中缀转后缀算法其主函数流程图为:返回计算结果图1 主函数算法流程图中缀转后缀算法流程图如下:放入exp数组中图2 中缀转后缀算法流程图计算后缀表达式流程图如下:图3 后缀表达式计算流程图第二种算法:两个栈算法其主函数流程图为:返回计算结果图4 主函数算法流程图直接计算数学表达式流程图如下:图5 直接计算表达式流程图三、源代码下面给出的是用中缀转后缀算法实现的程序的源代码:#include<stdio.h>#include<string.h>#include<math.h>#include<stdlib.h>#define MAXSIZE 100 //定义宏,数组最大长度为100//函数实现中缀转后缀,将存储数学表达式的数组str传参进来,exp存储后缀表达式void trans(char str[],char exp[]){struct{char data[MAXSIZE];//用来存放操作符int top;//数组下标}op;//用结构体创建操作符栈char ch;int i=0,j=0,tempi=0;op.top=-1;//给操作符栈初始化,令下标为-1while(ch!='\0'){ch=str[i]; //取str数组的第i个元素赋值给chif((ch>='0'&& ch<='9') || ch=='.')//对数值操作{tempi=i;//若ch为数字或小数点,将其下标值赋给临时下标tempi//依次向后扫描str数组,若一直为数字,跳入while循环while((ch>='0' && ch<= '9') || ch == '.'){tempi++;exp[j]=ch;//将数字存入exp数组中j++;ch=str[tempi];//取str数组中下标为tempi的元素赋给ch}exp[j]='#';j++;//用#做分隔符,将数值分隔开i=tempi;//跳出循环,将此时的tempi赋给i,继续向后扫描}//对操作符操作else if(ch=='+'||ch=='-'||ch=='*'||ch=='/'||ch=='%' || ch == '(' || ch == ')'){int level(char op);//声明level函数if(ch=='(')//如果为(,直接进栈{op.top++;op.data[op.top]=ch;//进栈操作}else if(ch==')'){//如果为),一直出栈直到遇到(while(level(op.data[op.top])!=-1)//若栈顶元素不为(,进入while循环{exp[j]=op.data[op.top];//操作符出栈,存入exp数组中op.top--;j++;if(op.top==-1)break;//如果栈为空,跳出循环}op.top--;//左括号pop出来}else if(op.top==-1)//如果栈为空,直接进栈{op.top++;op.data[op.top]=ch;//进栈操作}//如果所扫描的操作符优先等级比栈顶元素高,直接进栈else if(level(ch)>level(op.data[op.top])){op.top++;op.data[op.top]=ch;//进栈操作}else{//如果所扫描的操作符优先等级没有栈顶元素高,//一直出栈直到比栈顶元素优先级高while(level(ch)<=level(op.data[op.top])){exp[j]=op.data[op.top];//出栈存入exp数组中op.top--;j++;if(op.top==-1)break;//如果栈为空,跳出循环}op.top++;op.data[op.top]=ch;//比栈顶元素优先级高,入栈}i++;//str下标加1,向后扫描}}while(op.top!=-1)//扫描结束后如果操作符栈不为空,出栈直至为空{exp[j]=op.data[op.top];//出栈存入exp数组中op.top--;j++;}exp[j]='\0';//赋\0结束exp字符数组}int level(char op)//判断操作符优先等级{if(op == '+' || op == '-')//若为+、-,等级为1return 1;else if(op == '*' || op == '/' || op == '%')return 2; //若为*、/、%,等级为2else if(op == '(')return -1 ; //若为(,等级为-1elsereturn -3; //其他等级为-3;}double calvalue(double od1,double od2,char tempop)//计算{switch(tempop){case '+':return od1 + od2; //计算加法case '-':return od1 - od2;//计算减法case '*':return od1 * od2;//计算乘法case '/':return od1 / od2;//计算除法case '%':return fmod(od1,od2);//求余}return 0;}double calculate(char exp[])//计算后缀表达式{struct //用结构体创建值栈{double data[MAXSIZE]; //存储数值int top;}od;double d; //声明d变量存储数值double od1,od2; //存储值栈依次pop出来的操作数char ch;char tempch[20]; //声明临时数组存储子串int j=0,t;int length=strlen(exp);//计算exp数组的长度od.top=-1; //初始化值栈,令下标为-1while(j<length){ch=exp[j];//提取exp中第j个元素if(ch!='+' && ch!='-' && ch!= '*' && ch!='/' && ch!='%'){//如果为数字或小数点t=0;while((ch>='0' && ch<='9') ||ch=='.'){tempch[t]=ch;t++;//依次存放到临时数组中j++;ch=exp[j];}tempch[t]='\0';//结束tempch数组d=atof(tempch);//将子串转化成double类型的数od.top++;od.data[od.top]=d;//入值栈}else //若为操作符,从值栈中pop出两个数计算{od2=od.data[od.top];od.top--;//先出栈的赋给od2od1=od.data[od.top]; //后出栈的赋给od1od.data[od.top]=calvalue(od1,od2,ch); //计算出结果后再入栈}j++;}return od.data[od.top];//将结束后值栈中的数pop出来,即为计算结果}main(){char str[MAXSIZE],exps[MAXSIZE]; //定义两个数组printf("请输入算术表达式:\n");gets(str); //从控制台输入算数表达式printf("表达式为:%s\n",str);trans(str,exps); //调用trans函数,得到后缀表达式printf("后缀表达式:%s\n",exps);printf("结果为:%lf\n", calculate(exps)); //调用calculate函数,计算结果}下面给出的是用两个栈算法实现的程序的源代码:#include<stdio.h>#include<string.h>#include<math.h>#include<stdlib.h>#define MAXSIZE 100 //定义宏,数组最大长度为100double calculate(char str[]){struct //用结构体创建操作符栈{char data[MAXSIZE];//用来存放操作符}op;struct //用结构体创建值栈{double data[MAXSIZE];//用来存放操作数int top;}od;char ch;char tempch[20];//声明临时数组存储子串int j=0,t;double d;double od1,od2;//存储值栈依次pop出来的操作数char tempop;int length=strlen(str);//计算str数组的长度op.top=-1;//初始化操作符栈,令下标为-1od.top=-1;//初始化值栈while(j<length){ch=str[j];if((ch>='0' && ch<='9') ||ch=='.')//若为数值或小数点{d=0;t=0;while((ch>='0' && ch<='9') ||ch=='.')//截取子串{tempch[t]=ch;t++;//赋值给临时数组j++;ch=str[j];}tempch[t]='\0';d=atof(tempch);//将子串转化成double类型的数od.top++;od.data[od.top]=d;//入值栈}//对操作符操作else if(ch=='+'||ch=='-'||ch=='*'||ch=='/'||ch=='%' || ch == '(' || ch == ')') {if(ch=='(')//如果为(,直接进栈{op.top++;op.data[op.top]=ch;//进栈操作}else if(ch==')')//如果为),一直出栈直到遇到({int level(char op);//声明level函数while(level(op.data[op.top])!=-1)//若栈顶元素不为(,进入while循环{//声明calvalue函数double calvalue(double od1,double od2,char tempop);od2=od.data[od.top];od.top--;od1=od.data[od.top];tempop=op.data[op.top];op.top--;od.data[od.top]=calvalue(od1,od2,tempop);//计算出结果后入值栈if(op.top==-1)break;//如果操作符栈为空,跳出循环}op.top--;//左括号pop出来}else if(op.top==-1)//如果栈为空,直接进栈{op.top++;op.data[op.top]=ch;//进栈操作}//如果所扫描的操作符优先等级比栈顶元素高,直接进栈else if(level(ch)>level(op.data[op.top])){op.top++;op.data[op.top]=ch;//进栈操作}else{//如果所扫描的操作符优先等级没有栈顶元素高,//一直出栈直到比栈顶元素优先级高while(level(ch)<=level(op.data[op.top])){od2=od.data[od.top];od.top--;od1=od.data[od.top];tempop=op.data[op.top];op.top--;od.data[od.top]=calvalue(od1,od2,tempop);//计算结果后入值栈if(op.top==-1)break;//如果栈为空,跳出循环}op.top++;op.data[op.top]=ch;//比栈顶元素优先级高,入操作符栈}j++;//str下标加1,向后扫描}}while(op.top!=-1)//扫描结束后如果操作符栈不为空,出栈直至为空{od2=od.data[od.top];od.top--;od1=od.data[od.top];tempop=op.data[op.top];op.top--;od.data[od.top]=calvalue(od1,od2,tempop);//计算结果后入值栈}return od.data[od.top];//将结束后值栈中的数pop出来,即为计算结果}int level(char op)//判断操作符优先等级{if(op == '+' || op == '-')//若为+、-,等级为1return 1;else if(op == '*' || op == '/' || op == '%')return 2; //若为*、/、%,等级为2else if(op == '(')return -1 ; //若为(,等级为-1elsereturn -3; //其他等级为-3;}double calvalue(double od1,double od2,char tempop)//计算{switch(tempop){case '+':return od1 + od2;//计算加法case '-':return od1 - od2;//计算减法case '*':return od1 * od2;//计算乘法case '/':return od1 / od2;//计算除法case '%':return fmod(od1,od2);//求余}return 0;}void main(){char str[MAXSIZE];//定义str数组存放数学表达式printf("输入算术表达式:\n");gets(str); //从控制台输入算数表达式printf("结果是:%lf\n",calculate(str));//调用calculate函数,计算结果}四、运行结果图6 中缀转后缀算法运行结果图7 两个栈算法运行结果五、遇到的问题及解决编程的前期工作很重要,需要明确的理清思路,而编写运行的过程中更是会出现很多问题,有因粗心造成的拼写错误,有语法错误,也有逻辑错误。

相关文档
最新文档