一起talk C栗子吧(第二十一回:C语言实例--表达式求值)

2024-03-12 05:18

本文主要是介绍一起talk C栗子吧(第二十一回:C语言实例--表达式求值),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


各位看官们,大家好,前几回中咱们说了堆栈的原理,并且举了实际的例子进行解说,这一回咱们说的例

子是:表达式求值。表达式求值和上一回中说的括号匹配一样,都使用了堆栈的原理,大家可以从例子中

看出来,所以我们把它们放在一起。闲话休提,言归正转。让我们一起talk C栗子吧!


看官们,我们在这里说的表达式为包含加,减,乘除的四则运算表达式。例如:1+2*3-4/5就是一个四则运

算表达式。这个表达式中,运算符在数字中间,所以我们叫它中缀表达式,这也是符合我们思维的一种表

现形式,不过,计算机就不理解中缀表达式了,因为它没有办法像我们一样先进行乘除运算,然后再进行

加减运算,所以我们需要把中缀表达式转换成计算机能理解的后缀表达式。“什么是后缀表达式”,这时候

台下有看官在提问了,看官莫急,所谓的后缀表达式就是运算符在数字后面。例如:" 1+2*3-6/3 "这个中缀

表达式可以为" 123*+63/- "这种后缀表达式.


表达式求值有两个大的步骤:

  • 中缀表达式转换为后缀表达式。
  • 对后缀表达式进行求值。

这两个大的步骤中还有一些小的步骤,接下来我们详细说说这些小步骤。在说之前,我首先说明一个概念:优

先级。优先级代表着先后顺序,举一个日常为生活中的例子:排队买东西的时候,排在队列前面的人,比排在

队列后面人具有优先买东西的权利,我们就可以说:排在队列前面的人买东西的优先级高。优先级在表达式运

算过程中体现为:乘法和除法的优先级比加法和减法的优先级高。也就是我们通常说的先乘除后加减,这个内

容我就不多说了,大家在小学数学中都学过。我们在表达式求值过程中把中缀表达式转换为后缀表达式也与优

先级有关,因为后缀表达式可以去掉运算符的优先级。没有优先级了,计算机就能理解后缀表达式并对其进行

相关的运算。


中缀表达式转换为后缀表达式的步骤如下:

1.从头到尾依次遍历中缀表达式,每次从表达式中读取一个字符;

2.判断步骤1中读取的字符,如果是数字则保存到数组a中,如果是+*等运算符,请看下一个步骤;

3.对存放运算符的栈进行出栈操作,把步骤的2中的运算符和刚才出栈的运算符进行优先级比较;

4.如果步骤2中的运算符优先级高,那么把参与比较的这两个运算符都入栈。否则看下一步;

5.如果步骤2中的运算符优先级低,那么让栈中的运算符继续出栈,并且把出栈的运算符存放数组a中;

6.重复步骤4和步骤5,直到出栈运算符的优先级比步骤2中运算符的优先级高为止;

7.重复步骤1到步骤6.直到遍历完中缀表达式为止;

8.判断栈中是否还有运算符,如果有的话,就把所有运算符出栈,并且把出栈的运算符存放数组a中。


对后缀表达式求值的步骤如下:

1.从头到尾依次遍历后缀表达式,每次从表达式中读取一个字符;

2.判断步骤1中读取的字符,如果是数字则入栈,如果是+*等运算符,请看下一个步骤;

3.对存放数字的栈进行两次出栈操作,依据步骤2中运算符的类型,对出栈的两个数字进行运算;

4.对步骤3中的运算结果进行入栈操作,这样可以把运算结果保存到存放数字的栈中;

5.重复步骤1到步骤4.直到遍历完后缀表达式为止;

6.栈中最后一个元素就是该表达式的运算结果。


看官们,正文中就不写代码了,详细的代码放到了我的资源中,大家可以点击链接下载使用。从代码中可

以看到,我们用了两次栈,一次是在中缀表达式转换成后缀表达式的过程中,栈用来存放运算符。另外一

次是在后缀表达式求值的过程中,栈用来存放参与运算的数字。


各位看官,关于表达式求值的例子咱们就说到这里。欲知后面还有什么例子,且听下回分解。


这篇关于一起talk C栗子吧(第二十一回:C语言实例--表达式求值)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/800236

相关文章

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C语言进阶(预处理命令详解)

《C语言进阶(预处理命令详解)》文章讲解了宏定义规范、头文件包含方式及条件编译应用,强调带参宏需加括号避免计算错误,头文件应声明函数原型以便主函数调用,条件编译通过宏定义控制代码编译,适用于测试与模块... 目录1.宏定义1.1不带参宏1.2带参宏2.头文件的包含2.1头文件中的内容2.2工程结构3.条件编

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

MySQL多实例管理如何在一台主机上运行多个mysql

《MySQL多实例管理如何在一台主机上运行多个mysql》文章详解了在Linux主机上通过二进制方式安装MySQL多实例的步骤,涵盖端口配置、数据目录准备、初始化与启动流程,以及排错方法,适用于构建读... 目录一、什么是mysql多实例二、二进制方式安装MySQL1.获取二进制代码包2.安装基础依赖3.清

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束

Apache Ignite缓存基本操作实例详解

《ApacheIgnite缓存基本操作实例详解》文章介绍了ApacheIgnite中IgniteCache的基本操作,涵盖缓存获取、动态创建、销毁、原子及条件更新、异步执行,强调线程池注意事项,避免... 目录一、获取缓存实例(Getting an Instance of a Cache)示例代码:二、动态

Go语言编译环境设置教程

《Go语言编译环境设置教程》Go语言支持高并发(goroutine)、自动垃圾回收,编译为跨平台二进制文件,云原生兼容且社区活跃,开发便捷,内置测试与vet工具辅助检测错误,依赖模块化管理,提升开发效... 目录Go语言优势下载 Go  配置编译环境配置 GOPROXYIDE 设置(VS Code)一些基本

JSONArray在Java中的应用操作实例

《JSONArray在Java中的应用操作实例》JSONArray是org.json库用于处理JSON数组的类,可将Java对象(Map/List)转换为JSON格式,提供增删改查等操作,适用于前后端... 目录1. jsONArray定义与功能1.1 JSONArray概念阐释1.1.1 什么是JSONA

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串