当前位置: 首页 > wzjs >正文

网站做产品的审核工作内容360站长平台

网站做产品的审核工作内容,360站长平台,律师事务所网站建设方案,wordpress 文章浏览量二叉树的遍历前序遍历中序遍历后续遍历总结 二叉树的遍历 虽然用递归的方法遍历二叉树实现起来更简单,但是要想深入理解二叉树的遍历,我们还必须要掌握用栈遍历二叉树,递归其实就是利用了系统栈去遍历。特此记录一下如何用双重视角去看待二叉…

  • 二叉树的遍历
    • 前序遍历
    • 中序遍历
    • 后续遍历
    • 总结

二叉树的遍历

虽然用递归的方法遍历二叉树实现起来更简单,但是要想深入理解二叉树的遍历,我们还必须要掌握用栈遍历二叉树,递归其实就是利用了系统栈去遍历。特此记录一下如何用双重视角去看待二叉树的遍历,加深一下理解。

前序遍历

我们从前序遍历入手,搞懂了一个,其它的也就容易了。

使用递归的方法遍历的话很简单,代码如下:

public void preorderTraversal(TreeNode root, List<Integer> result) {if (root == null) {return;}// 访问根节点result.add(root.val);// 递归遍历左子树preorderTraversal(root.left, result);// 递归遍历右子树preorderTraversal(root.right, result);
}

它利用了系统中线程的栈空间,先访问当前节点,再调用自身方法递归地去对左子节点进行前序遍历,这在线程栈空间中会新增加一个方法。线程也会优先去处理在线程栈上新增的方法。如下图所示:
在这里插入图片描述
这里发生的事情是:

  1. 系统为每次方法调用维护一个执行上下文,包含局部变量和返回地址
  2. 当执行到preorder(root.left)时,当前方法的执行被暂停,其状态被保存在系统栈中
  3. 系统转而执行左子树的遍历,完成后会自动返回到保存的执行点
  4. 继续执行preorder(root.right)

关键点是:系统栈自动保存了"接下来要做什么"的信息。每个节点被处理时,系统知道处理完左子树后还需要回来处理右子树。

而使用栈的话,我们只需要按照递归遍历的方式自己创建一个栈模拟着线程栈的方法去遍历就行。代码如下:

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;public List<Integer> preorderTraversal(TreeNode root) {List<Integer> result = new ArrayList<>();if (root == null) return result;Stack<TreeNode> stack = new Stack<>();stack.push(root);while (!stack.isEmpty()) {TreeNode node = stack.pop();result.add(node.val);// 先右后左入栈,这样出栈时会先处理左子节点if (node.right != null) {stack.push(node.right);}if (node.left != null) {stack.push(node.left);}}return result;
}

这里的关键区别是:

  1. 我们只能用栈存储节点本身,而不能存储"执行到哪一步"的完整上下文
  2. 栈顶元素是下一个要处理的节点,而不是一个带有执行状态的方法调用
  3. 由于栈的后进先出特性,为了先处理左子节点,必须先将右子节点入栈,再将左子节点入栈

系统线程栈和手动实现栈的区别

  1. 系统线程栈的一个栈帧的运行(不包括递归调用产生的跳转)等同于手动实现栈的三件事:1.访问当前节点 2.入栈右子节点 3.入栈左子节点
  2. 系统线程栈新增了一个栈帧等同于手动实现栈的:出栈一个节点作为当前节点

中序遍历

递归代码如下:

public void inorderTraversal(TreeNode root, List<Integer> result) {if (root == null) {return;}// 递归遍历左子树inorderTraversal(root.left, result);// 访问根节点result.add(root.val);// 递归遍历右子树inorderTraversal(root.right, result);
}

系统线程栈中一个栈帧的执行过程(不包括递归调用产生的跳转)等同于手动实现栈中的三个操作:

  1. 递归访问左子树
  2. 访问当前节点
  3. 递归访问右子树

手动实现栈代码如下:

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;public List<Integer> inorderTraversal(TreeNode root) {List<Integer> result = new ArrayList<>();Stack<TreeNode> stack = new Stack<>();TreeNode current = root;while (current != null || !stack.isEmpty()) {// 将所有左子节点入栈while (current != null) {stack.push(current);current = current.left;}// 处理栈顶节点current = stack.pop();result.add(current.val);// 转向右子树current = current.right;}return result;
}

核心思路是先将当前节点及其所有左子节点入栈,然后访问节点值,再处理右子树
无法用简单的"入栈-出栈-访问"模式表达,需要维护一个current指针跟踪当前处理节点
关键步骤:

  1. 将当前节点及其所有左子节点入栈
  2. 弹出栈顶节点并访问
  3. 将当前节点切换到右子节点,重复步骤1

后续遍历

递归代码如下:

public void postorderTraversal(TreeNode root, List<Integer> result) {if (root == null) {return;}// 递归遍历左子树postorderTraversal(root.left, result);// 递归遍历右子树postorderTraversal(root.right, result);// 访问根节点result.add(root.val);
}

系统线程栈中一个栈帧的执行过程等同于手动实现栈中的三个操作:

  1. 递归访问左子树
  2. 递归访问右子树
  3. 访问当前节点

手动实现栈代码如下:

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;public List<Integer> postorderTraversal(TreeNode root) {List<Integer> result = new ArrayList<>();if (root == null) return result;Stack<TreeNode> stack1 = new Stack<>();Stack<TreeNode> stack2 = new Stack<>();stack1.push(root);// 先将节点按 根-右-左 的顺序放入栈2while (!stack1.isEmpty()) {TreeNode node = stack1.pop();stack2.push(node);if (node.left != null) {stack1.push(node.left);}if (node.right != null) {stack1.push(node.right);}}// 从栈2中弹出的顺序就是 左-右-根while (!stack2.isEmpty()) {result.add(stack2.pop().val);}return result;
}

双栈法
系统线程栈中一个栈帧的执行等同于:

  1. 将当前节点压入第二个栈(结果栈)
  2. 将左子节点压入第一个栈(处理栈)
  3. 将右子节点压入第一个栈(处理栈)

注意:入栈顺序是"左-右",这样处理顺序变成"右-左",最终从结果栈弹出时顺序为"左-右-根"

单栈法

  1. 需要额外记录上次访问的节点
  2. 核心思路是判断右子树是否已访问,决定是访问节点还是处理右子树
  3. 由于逻辑较复杂,难以用简单的等价操作表述

总结

读者可以根据前序遍历的思路自行去理解中序遍历和后序遍历。重点就是理解系统的线程栈是怎么运作的,以及手动实现的栈是如何保存节点的。搞清楚了这两点,对二叉树的遍历的理解就会更上一层了。

http://www.dtcms.com/wzjs/49575.html

相关文章:

  • 全国房产查询系统西安seo阳建
  • 上海工商网上办事平台网站怎么优化关键词
  • 中国在菲律宾做网站百度pc端首页
  • 自己 做 网站天津seo排名公司
  • 亳州做商标网站的公司快速开发网站的应用程序
  • 营销推广型网站公司uc浏览网页版进入
  • 高端网站的建设站长之家seo查询
  • 做网站用服务器网络推广员每天的工作是什么
  • wordpress怎么让文章只显示摘要seo优化排名经验
  • 东莞国药官网网上商城郑州官网网站推广优化
  • 公司微网站建设方案seo怎么弄
  • 免费在线做网站seo流量增加软件
  • 茂县建设局网站网络营销推广有效方式
  • 怎样提高网站的打开速度网店如何引流与推广
  • 企业信用修复单页网站怎么优化
  • 南昌网站空间教育机构排名
  • 网站开发与app差距关键词查询工具
  • 中邮保险网站关键词有几种类型
  • 网站是专门对生活中的一些所谓常识做辟谣的深圳网络品牌推广公司
  • 安徽省建设工程信网站百度人工服务热线电话
  • 如何做视频播放网站北京网络推广公司排行
  • 自己家里做网站网速慢网址怎么推广
  • 企业网站类型主要包括互动营销公司
  • 邯郸做网站服务商如何增加网站的外链
  • 徐州市做网站图片优化软件
  • 湖北孝感展示型网站建设价格aso优化服务
  • 连云港网站建设培训学校郑州网络推广效果
  • 企业网站哪家做得好淘宝怎样优化关键词
  • 杭州 企业 建网站销售网站排名
  • 视频网站建设要多少钱seo排名软件怎么做