二叉树的非递归遍历

二叉树的非递归遍历

大家好,又见面了,我是全栈君。

先写下这个问题的模式

def preorderTraversal(self, root):
	if root == None: return []
	re = []
	insert root to stack s
	while s not empty:
		cur_root = top of stack s
		s.pop()
		how to handle cur_root
		how to handle cur_root.left
		how to handle cur_root.right

首先我们要把非空的root节点入栈,在循环里不断的推断出栈,然后处理栈顶元素及左孩子结点和右孩子结点。

我们先看下当前的栈顶元素怎么处理。先根遍历的顺序是:【根 左 右】。而我们每次处理的栈顶元素的身份事实上都恰好是根。所以我们就能够直接把这个根几点输出或放入结果容器中;那么其左孩子和右孩子怎样处理呢?既然是模拟递归,那么肯定要入栈进行保存的,谁先入栈呢?考虑到栈的性质,我们应该让其右孩子先入栈,左孩子后入栈,这样,栈顶就是左孩子,下次先出栈的就是左孩子。这样就符合先根遍历的顺序了。

再回过头来看下在左右孩子没入栈之前。我们不过获得了栈顶元素。该元素还依旧在栈中,那么它在栈中还有意义吗?非常明显没有意义了,由于它的信息我们已经输出,而其左右孩子在入栈后就再也不须要它了,所以就应该在左右孩子入栈前将其pop掉。

	def preorderIter(self, root):
		if None == root: return []
		re = []; s = []
		s.append(root)

		while len(s):
			cur_root = s.pop()
			print cur_root.val
			re.append(cur_root.val)

			if cur_root.right:
				s.append(cur_root.right)
			if cur_root.left:
				s.append(cur_root.left)
		return re

相同。后根遍历也是如此。尽管后根的遍历是【左 右 根】,可是我们毕竟是知道根要放在哪里。差别就是子节点的入栈顺序的变化。

	def postorderIter(self, root):
		if None == root:
			return
		re = [] # store results
		s = [] # node stack
		s.append(root)
		while len(s):
			cur_root = s.pop()
			re.insert(0,cur_root.val)
			if cur_root.left:
				s.append(cur_root.left)
			if cur_root.right:
				s.append(cur_root.right)
		return re

麻烦点的应该是中根遍历【左 根 右】。如前所述,我们处理的当前栈顶节点是视为根结点的,可是这个根结点却不知该放在结果中的哪里,放在前面,前面应该是左的位置。放在右面,右边应该是右孩子的位置,放中间?哪里算中间?1-10, 2 是中间还是3是中间?我们不确定。由于左右孩子的个数我们无从得知。

看来此时的栈顶元素不能像先根和后根那样。直接输出,还得在栈里面挤一下才好,不然总不能直接丢弃。

之所以当前的栈顶根不能输出,是由于它的左还没有确定,那么我们仅仅要把它的左都输出了。就能够确定当前根的位置了。

def inorderIter(self, root):
		if None == root: return []
		re = []
		s = []
		s.append(root)
		while len(s):
			cur_root = s[-1]
			# push, until the last letf
			while cur_root.left:
				s.append(cur_root.left)
				cur_root = cur_root.left
			# pop, until one node has right
			while len(s):
				cur_root = s[-1]
				print cur_root.val
				re.append(cur_root.val)
				s.pop()
				if cur_root.right:
					s.append(cur_root.right)
					break
		return re

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/115871.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • pycharm企业版激活码【2022最新】2022.01.22

    (pycharm企业版激活码)好多小伙伴总是说激活码老是失效,太麻烦,关注/收藏全栈君太难教程,2021永久激活的方法等着你。https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~1TCF2R91JZ-eyJsaWNlbnNlSWQiOi…

    2022年3月31日
    86
  • android listview单击事件

    android listview单击事件今天我们来学习下listview单击事件,这在开发中是经常用的组件之一。1.新建一个项目,名为ListViewDemo。2.布置布局文件main.xmlandroid:orientation=”vertical”android:layout_width=”fill_parent”android:layout_height=”fill_parent”>

    2022年7月22日
    10
  • 怎么判断值、对象等是否为空

    怎么判断值、对象等是否为空

    2021年9月3日
    58
  • java 常量池和运行时常量池_常量池中的字符串是对象吗

    java 常量池和运行时常量池_常量池中的字符串是对象吗简介:这几天在看Java虚拟机方面的知识时,看到了有几种不同常量池的说法,然后我就去CSDN、博客园等上找资料,里面说的内容真是百花齐放,各自争艳,因此,我好好整理了一下,将我自认为对的理解写下来与大家共同探讨:在Java的内存分配中,总共3种常量池:1.字符串常量池(StringConstantPool):1.1:字符串常量池在Java内存区域的哪个位置?在JDK6.0及之前版本,字符串

    2022年7月28日
    5
  • html遮罩层动画制作,flash简单制作遮罩动画效果[通俗易懂]

    html遮罩层动画制作,flash简单制作遮罩动画效果[通俗易懂]flash简单制作遮罩动画效果QQ空间的开机动画大家应该都有,从最初的出现的一点到后面全部出现,如此神奇的效果到底是怎么做的呢,一起来看看吧!遮罩特效:由于百度只能上传500k以内的照片,所以效果图片质量不是很好,当然,我们一般做的特效是.swf,这里是为了方便大家观看,所以做成了gif.步骤:1、打开flash面板,创建新项目。2、点击文件——导入——导入一张图片。3、窗口——库,这里我们可以…

    2022年5月1日
    93
  • c语言中的offset_c语言中/和%的区别

    c语言中的offset_c语言中/和%的区别今天看libPhenom源代码,看到他们使用的JSON解析库参考的是JanssonJSON解析库。于是就去网上查了这个库,找到了官方网站:http://www.digip.org/jansson/。找了一下发现在Github上能够下载源代码,于是下载了源代码来瞅瞅。    看了一会儿发现有一块代码一直看不明白,就比如说如下的代码:json_t*json_object(void)

    2022年8月22日
    5

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号