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

LeetCode经典算法题:二叉树遍历(递归遍历+迭代遍历+层序遍历)以及线索二叉树java详解

LeetCode经典算法题:二叉树遍历(递归遍历+迭代遍历+层序遍历)以及线索二叉树java详解

文章目录

  • 二叉树遍历
    • 题目描述
    • 解题思路与代码
      • 递归遍历
      • 迭代遍历
      • 层序遍历
    • 线索二叉树:

二叉树遍历

题目描述

从根节点往下查找,先找左子树、直至左子树为空(左子节点逐个入栈、直至左子节点为空),再找右子树(出栈找右子节点)
前序遍历:根左右,第一次经过节点即打印,直到打印null,往回溯,打印右子树
中序遍历:左根右,第二次经过该节点时进行打印,即左边回溯时
后序遍历:左右根,第三次经过该节点时进行打印,即右边回溯时
层序遍历:按照层级,从上往下,从左到右。使用广度优先搜索算法。
从根节点往下查找,先找左子树、直至左子树为空(左子节点逐个入栈、直至左子节点为空),再找右子树(出栈找右子节点)

解题思路与代码

递归遍历

    public static void preorder(TreeNode root) {if (root == null) {return;}//System.out.println(root.val);//前序 第一次成为栈顶preorder(root.left);System.out.println(root.val);//中序 第二次成为栈顶preorder(root.right);//System.out.println(root.val);//后序 第三次成为栈顶}

迭代遍历

    //前序:使用stack记录递归路径,左子节点后添加保证先出栈public static void preOrder2(TreeNode head) {if (head != null) {Stack<TreeNode> stack = new Stack<TreeNode>();stack.add(head);while (!stack.isEmpty()) {head = stack.pop();if(head != null){System.out.println(head.val);stack.push(head.right);stack.push(head.left);}}}}//中序:将左子节点入栈,出栈打印值,然后添加右子节点public static void preOrder3(TreeNode head) {if (head != null) {Stack<TreeNode> stack = new Stack<TreeNode>();while (!stack.isEmpty() || head != null) {if (head != null) {stack.push(head);head = head.left;} else {head = stack.pop();System.out.println(head.val);head = head.right;}}}}//后序:public static void postorderTraversal(TreeNode root) {if (root == null) {return ;}Deque<TreeNode> stack = new LinkedList<TreeNode>();TreeNode prev = null;while (root != null || !stack.isEmpty()) {while (root != null) {stack.push(root);root = root.left;}root = stack.pop();//root的左子节点为nullif (root.right == null || root.right == prev) {//右子节点为null,或者右子节点已打印System.out.println(root.val);prev = root;root = null;} else {//右子节点有值,重新入栈stack.push(root);root = root.right;}}}

层序遍历

   public static void levelTraversal(Node root) {Queue<Node> q = new LinkedList<>();q.add(root);while (!q.isEmpty()) {Node temp = q.poll();if (temp != null) {System.out.print(temp.value + " ");q.add(temp.left);q.add(temp.right);}}}public static void deepOrder(TreeNode root) {if (root == null) {return ;}Queue<TreeNode> queue = new LinkedList<TreeNode>();queue.offer(root);while (!queue.isEmpty()) {for (int i = 1; i <= queue.size(); ++i) {TreeNode node = queue.poll();System.out.println(node.val);if (node.left != null) {queue.offer(node.left);}if (node.right != null) {queue.offer(node.right);}}}}private static List order(TreeNode root, int i, ArrayList list) {if (root == null) {return null;}int length = list.size();if(length <= i){for(int j=0; j<= i-length; j++){list.add(length+j,null);}}list.set(i,root.val);order(root.left, 2 * i,list);order(root.right, 2 * i + 1,list);return list;}

线索二叉树:

在N个节点的二叉树中,每个节点有2个指针,所以一共有2N个指针,除了根节点以外,每一个节点都有一个指针从它的父节点指向它,所以一共使用了N-1个指针,所以剩下2N-(N-1)也就是N+1个空指针;

如果能利用这些空指针域来存放指向该节点的直接前驱或是直接后继的指针,则可由此信息直接找到在该遍历次序下的前驱节点或后继节点,从而比递归遍历提高了遍历速度,节省了建立系统递归栈所使用的存储空间;

这些被重新利用起来的空指针就被称为线索(Thread),加上了线索的二叉树就是线索二叉树实现思路:按某种次序遍历二叉树,在遍历过程中用线索取代空指针即可。以中序遍历为例,首先找到中序遍历的开始节点,然后利用线索依次查找后继节点即可。

由于它充分利用了空指针域的空间(等于节省了空间),又保证了创建时的一次遍历就可以终生受用前驱、后继的信息(这意味着节省了时间),所以在实际问题中,如果所使用的二叉树需要经常遍历或查找节点时需要某种遍历中的前驱和后继,那么采用线索二叉链表的存储结构就是不错的选择morris遍历:构建中序线索二叉树的过程中,如果发现前驱节点的右指针指向自身,则将指针(线索)删除

public static void morrisPre(Node cur) {if(head == null){return;}Node mostRight = null;while (cur != null){// cur表示当前节点,mostRight表示cur的左孩子的最右节点mostRight = cur.left;if(mostRight != null){// cur有左孩子,找到cur左子树最右节点while (mostRight.right !=null && mostRight.right != cur){mostRight = mostRight.right;}// mostRight的右孩子指向空,让其指向cur,cur向左移动if(mostRight.right == null){mostRight.right = cur;System.out.print(cur.value+" ");cur = cur.left;continue;}else {// mostRight的右孩子指向cur,让其指向空,cur向右移动mostRight.right = null;}}else {System.out.print(cur.value + " ");}cur = cur.right;}}public static void morrisIn(Node cur) {if(head == null){return;}Node mostRight = null;while (cur != null){mostRight = cur.left;if(mostRight != null){while (mostRight.right !=null && mostRight.right != cur){mostRight = mostRight.right;}if(mostRight.right == null){mostRight.right = cur;cur = cur.left;continue;}else {mostRight.right = null;}}System.out.print(cur.value+" ");cur = cur.right;}}public static void morrisPos(TreeNode cur) {if (cur == null) {return;}TreeNode head = cur;TreeNode mostRight = null;while (cur != null) {mostRight = cur.left;if (mostRight != null) {while (mostRight.right != null && mostRight.right != cur) {mostRight = mostRight.right;}if (mostRight.right == null) {mostRight.right = cur;cur = cur.left;continue;} else {mostRight.right = null;printEdge(cur.left);}}cur = cur.right;}printEdge(head);System.out.println();}public static void printEdge(TreeNode head) {TreeNode tail = reverseEdge(head);TreeNode cur = tail;while (cur != null) {System.out.print(cur.val + " ");cur = cur.right;}reverseEdge(tail);}public static TreeNode reverseEdge(TreeNode from) {TreeNode pre = null;TreeNode next = null;while (from != null) {next = from.rightfrom.right = pre;pre = from;from = next;}return pre;}public List<Integer> postorderTraversal(TreeNode root) {List<Integer> res = new ArrayList<Integer>();if (root == null) {return res;}Deque<TreeNode> stack = new LinkedList<TreeNode>();TreeNode prev = null;while (root != null || !stack.isEmpty()) {while (root != null) {stack.push(root);root = root.left;}root = stack.pop();if (root.right == null || root.right == prev) {res.add(root.val);prev = root;root = null;} else {stack.push(root);root = root.right;}}return res;}
http://www.lryc.cn/news/15610.html

相关文章:

  • 【Java闭关修炼】MyBatis-接口代理的方式实现Dao层
  • 2022年网络安全政策态势分析与2023年立法趋势
  • 使用vmware制作云平台redhat7.9镜像模板
  • OpenCV基础(28)使用OpenCV进行摄像机标定Python和C++
  • APB总线详解及手撕代码
  • 【Linux/Windows】源文件乱码问题解决方法总结
  • Python 四大主流 Web 编程框架
  • 学UI设计,可以向哪些方向发展?该怎么学?
  • 【C++】初识CC++内存管理
  • Nacos快速使用指南
  • 复旦发布国内首个类ChatGPT模型MOSS,和《流浪地球》有关?
  • 国家级高新区企业主要经济指标(2012-2021年)
  • SpringBoot2核心技术-核心功能【05、Web开发】
  • 2021-03 青少年软件编程(C语言)等级考试试卷(六级)解析
  • 数据库的多租户隔离
  • 网络输入分辨率是否越大越好
  • 离线采集普遍解决方案
  • SAP ABAP 数据类型P类型详解
  • 应用沙盒seccomp的使用
  • C++项目——高并发内存池(2)——thread_cache的基础功能实现
  • 【C进阶】数据的存储
  • 【已解决】异常断电文件损坏clickhouse启动不了:filesystem error Structure needs cleaning
  • FlinkSQL行级权限解决方案及源码
  • 【基础篇】8 # 递归:如何避免出现堆栈溢出呢?
  • 基于微信公众号(服务号)实现扫码自动登录系统功能
  • AXI实战(二)-跟着产品手册设计AXI-Lite外设(AXI-Lite转串口实现)
  • 一周搞定模拟电路视频教程,拒绝讲PPT,仿真软件配合教学,真正一周搞定
  • 高德地图获得角度
  • 【C++】-- C++11基础常用知识点(下)
  • 提到数字化,你想到哪些关键词