• 表达式求值--堆栈应用


    理解对运算数和运算符的操作:

     重要的下面三个问题:

    • 中缀表达式如何直接求值?
    • 后缀表达式如何直接求值?
    • 中缀表达式如何转换为后缀表达式?

         将一个中序表达式转化成为逆波兰表达式的方法其实很简单,也是一个成型的算法。通过逆波兰表达式求一个计算式子的值,也是很简单的,只要遇到过会用就行了。

    1、将一个中序表达式转化成为逆波兰表达式。(逆波兰表达式又叫做后缀表达式)

           首先维护的是两个栈,我们这里暂且称为S1和S2,S1中的结果最后存的就是逆波兰表达式,S2中将用于暂时存放运算符并且在最终形成逆波兰表达式的时候,该栈是会清空的。下面我们看看怎样具体的形成逆波兰表达式。

           在此首先定义一下运算符的优先级关系,从小到达排序,相同优先级没有用逗号隔开:(,+-,*,负号,)。

           从左至右遍历一个给定的中序表达式,也就是我们常规的数学计算的表达式。

    (1)如果遇到的是数字,我们直接加入到栈S1中;

    (2)如果遇到的是左括号,则直接将该左括号加入到栈S2中;

    (3)如果遇到的是右括号,那么将栈S2中的运算符一次出栈加入到栈S1中,直到遇到左括号,但是该左括号出栈S2并不加入到栈S1中;

    (4)如果遇到的是运算符,包括单目运算符和双目运算符,我们按照下面的规则进行操作:

              (1)如果此时栈S2为空,则直接将运算符加入到栈S2中;

              (2)如果此时栈S2不为空,当前遍历的运算符的优先级大于等于栈顶运算符的优先级,那么直接入栈S2;

              (3)如果此时栈S2不为空,当前遍历的运算符的优先级小于栈顶运算符的优先级,则将栈顶运算符一直出栈加入到栈S1中,直到栈为空或者遇到一个运算符的优先级小于等于当前遍历的运算符的优先级,此时将该运算符加入到栈S2中;

    (5)直到遍历完整个中序表达式之后,栈S2中仍然存在运算符,那么将这些运算符依次出栈加入到栈S1中,直到栈为空。

           按照上面的五条操作反复进行完成,那么栈S1中存放的就是逆波兰表达式。

     

    2、利用逆波兰表达式求值

           利用逆波兰表达式求计算式的值其实很简单,正式因为这一点,所以逆波兰表达式才在编译原理中被用于计算一个表达式的值。

           下面来具体看看如何求一个逆波兰表达式的值:

           我们此时维护一个数据结果栈S3,我们将会看到该栈中最后存放的是最终的表达式的值。我们从左至右的遍历栈S1,然后按照下面的规则进行操作栈S3.

    (1)如果遇到的是数字,那么直接将数字压入到S3中;

    (2)如果遇到的是单目运算符,那么取S3栈顶的一个元素进行单目运算之后,将结果再次压入到栈S3中;

    (3)如果遇到的是双目运算符,那么取S3栈顶的两个元素进行,首先出栈的在左,后出栈的在右进行双目运算符的计算,将结果再次压入到S3中。

           按照上面的三个规则,遍历完整个栈S1,那么最后S3中的值就是逆波兰表达式的值了,所以我们可以看出来使用逆波兰表达式进行求值是很简单的,只有两种操作要么是直接压栈,要么是运算之后将结果压栈。

     

    reference:中缀表达式求值问题

     

  • 相关阅读:
    __setattr__,__getattr__,__delattr__
    LeetCode 面试题42. 连续子数组的最大和
    LeetCode 53. 最大子序和
    LeetCode 面试题39. 数组中出现次数超过一半的数字
    LeetCode 169. 多数元素
    LeetCode 426.将二叉搜索树转化为排序的双向链表
    LeetCode 面试题36. 二叉搜索树与双向链表
    LeetCode 面试题35. 复杂链表的复制
    LeetCode 138. 复制带随机指针的链表
    LeetCode 面试题34. 二叉树中和为某一值的路径
  • 原文地址:https://www.cnblogs.com/ranjiewen/p/6582019.html
Copyright © 2020-2023  润新知