二叉树与表达式(前缀,中缀,后缀表达式)

表达式简介
最近看书的时候接触到了后缀表达式,恰好看SICP中间也碰到了类似的问题,就花时间好好将这部分内容进行了整理。

前缀表达式(Prefix Notation)是指将运算符写在前面操作数写在后面的不包含括号的表达式,而且为了纪念其发明者波兰数学家Jan Lukasiewicz所以前缀表达式也叫做“波兰表达式”。比如- 1 + 2 3

后缀表达式(Postfix Notation)与之相反,是指运算符写在操作数后面的不含括号的算术表达式,也叫做逆波兰表达式。比如1 2 3 + -

中缀表达式(Infix Notation)就是常用的将操作符放在操作数中间的算术表达式。前缀表达式和后缀表达式相对于中缀表达式最大的不同就是去掉了表示运算优先级的括号,比如1-2+3

现在让我们先回想一下小学数学中关于中缀表达式求值的相关的规则:

1 有括号的情况下,先计算括号内,再计算括号外

2 在无括号的情况下,考虑运算的优先级,先乘除,后加减

3 同样优先级的情况下,从左至右进行计算

在中缀表达式的情况下求值,既要考虑括号,优先级,还要考虑操作出现的先后顺序。但是,作为计算机,其计算过程就显的比较复杂,对于一个中缀表达式,需要不停地对表达式进行多次遍历,来查找相应的计算的信息。这样从算法复杂度上来说,是不可取的。

前缀表达式和后缀表达式相对于人们常用的中缀表达式最大的不同就在于表达式中的运算符是按照一定的顺序出现(接下来会具体讲解),所以求值过程中并不需要在表达式中使用括号来指定运算顺序,也不需要在计算过程中其中考虑运算符号的优先级。在采用辅助数据结构的情况下,只需要对表达式进行一次遍历即可计算出结果,大大降低了算法复杂度,也更加符合传统计算机的工作方式。

表达式的转换
在对前缀中缀和后缀表达式进行了简单的介绍以后,接下来对表达式之间的转换进行一个简单的阐述,在这里首先以中缀表达式转换后缀表达式为例进行较详细的讲解。

首先介绍一个人工转换的方法,假设有一个中缀表达式a+b*c-(d+e)

1首先将这个中缀表达式的所有运算加括号((a+(b*c))-(d+e))

2然后将所有运算符放到括号后面,这样就变成了((a(bc)* )+ (de)+ )-

3把所有括号去掉abc*+de+-,最后得出的结果就是后缀表达式

上面这个方法可以在比如做题分析的时候用人脑的时候使用,接下来介绍用程序实现将中缀转换成后缀表达式的思路

1 建立一个栈,然后从左至右的遍历中缀表达式

2 如果碰到了操作数,则直接将其输出,如果碰到了操作符,这个时候则需要进行优先级的比较,如果栈为空或者栈顶的操作符的优先级比当前的低,则将当前的操作符Push入栈,如果栈顶操作符比当前的操作符的优先级高或者相同,则POP操作符并输出,直到出现前一种情况为止

3 需要注意的是对于括号需要另外注意一下,如果是’(’,则直接入栈,如果是)则要找到对应的’(’为止,并且当两者同时出现时则同时删除

下面模拟程序对a+b*c-(d+e)求中缀表达式,首先对其进行扫描

输出 a 栈底 栈顶

输出 a 栈底 + 栈顶

输出 a b 栈底 + 栈顶

输出 a b 栈底 + * 栈顶

输出 a b c 栈底 + * 栈顶

输出 a b c * 栈底 + 栈顶

输出 a b c * + 栈底 - 栈顶

输出 a b c * + 栈底 - ( 栈顶

输出 a b c * + d 栈底 - ( 栈顶

输出 a b c * + d 栈底 - ( + 栈顶

输出 a b c * + d e 栈底 - ( + 栈顶

输出 a b c * + d e + 栈底 ( 栈顶

输出 a b c * + d e + - 栈底 栈顶

上面已经把中缀表达式转换后缀表达式的过程讲解了一下,而中缀表达式转换前缀表达式其实过程和转换后缀表达式及其相似,在这里就不说具体过程了,只是对其中的几个不同之处稍微说明一下:

1 转换后缀表达式由于符号是要在操作数后面出现,所以操作数入栈,扫描顺序是从左向右,转换前缀表达式的话操作符需要在操作数前面出现,那么扫描顺序就应该是从右向左。

2 操作符的具体处理方式和转换到后缀表达式略有不同

3 括号的操作也是’)’直接入栈,’(’则配对’)’后一起出栈

前缀,中缀,后缀表达式与二叉树的表示
这三个表达式其实与二叉树是有这很紧密的联系的,a+b*c-(d+e)这个中缀表达式我们将其操作符当做内节点,操作数当做叶子节点,这样的话就能够画出这个中缀表达式所对应的二叉树

image

如果我们对这个中缀表达式进行前序和后序遍历,则输出的结果就是其对应的前缀和后缀表达式。

参考的资料

http://liuaigui.blog.sohu.com/40175057.html
http://apps.hi.baidu.com/share/detail/19581670
http://apps.hi.baidu.com/share/detail/19581670

转自博客园(详细出处不详)

  • 2
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
关于 Python 中的 `self` 关键字: 在 Python 中,`self` 是一个约定俗成的关键字,通常作为类方法的第一个参数出现。它表示类的实例对象本身,可以用来访问实例变量和方法。在 Python 中,不像其他语言中使用 `this` 或 `self` 关键字一样强制要求使用,但是为了代码的可读性和规范性,建议在类方法中使用 `self`。 关于 Python 实现中序遍历表达式二叉树前缀中缀后缀表达式生成表达式二叉树: 以下是一个示例代码,实现了中序遍历表达式二叉树前缀中缀后缀表达式生成表达式二叉树的功能: ```python class TreeNode: def __init__(self, val): self.val = val self.left = None self.right = None class ExpressionTree: def __init__(self): self.root = None # 中序遍历表达式二叉树 def inorder_traversal(self, node): if node.left: self.inorder_traversal(node.left) print(node.val, end=' ') if node.right: self.inorder_traversal(node.right) # 前缀表达式生成表达式二叉树 def build_from_prefix(self, expression): stack = [] for i in range(len(expression) - 1, -1, -1): if expression[i].isdigit(): node = TreeNode(expression[i]) stack.append(node) else: node = TreeNode(expression[i]) node.left = stack.pop() node.right = stack.pop() stack.append(node) self.root = stack.pop() # 中缀表达式生成表达式二叉树 def build_from_infix(self, expression): stack = [] i = 0 while i < len(expression): if expression[i].isdigit(): j = i while j < len(expression) and expression[j].isdigit(): j += 1 node = TreeNode(expression[i:j]) stack.append(node) i = j elif expression[i] == '(': stack.append('(') i += 1 elif expression[i] == ')': while stack[-1] != '(': right = stack.pop() op = stack.pop() left = stack.pop() node = TreeNode(op) node.left = left node.right = right stack.append(node) stack.pop() # 弹出左括号 i += 1 else: while stack and stack[-1] != '(' and self.precedence(stack[-1]) >= self.precedence(expression[i]): right = stack.pop() op = stack.pop() left = stack.pop() node = TreeNode(op) node.left = left node.right = right stack.append(node) stack.append(expression[i]) i += 1 while len(stack) > 1: right = stack.pop() op = stack.pop() left = stack.pop() node = TreeNode(op) node.left = left node.right = right stack.append(node) self.root = stack.pop() # 后缀表达式生成表达式二叉树 def build_from_postfix(self, expression): stack = [] for c in expression: if c.isdigit(): node = TreeNode(c) stack.append(node) else: right = stack.pop() left = stack.pop() node = TreeNode(c) node.left = left node.right = right stack.append(node) self.root = stack.pop() # 返回操作符优先级 def precedence(self, op): if op == '+' or op == '-': return 1 elif op == '*' or op == '/': return 2 else: return 0 ``` 其中,`TreeNode` 表示二叉树的节点,`ExpressionTree` 表示表达式二叉树。`build_from_prefix`、`build_from_infix`、`build_from_postfix` 分别表示通过前缀中缀后缀表达式生成表达式二叉树的方法。`inorder_traversal` 表示中序遍历表达式二叉树的方法。`precedence` 表示返回操作符优先级的方法。 示例代码中使用了栈来辅助实现表达式树的构建,具体的实现方法可以参考代码中的注释。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值