全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

java实现二叉树的创建及5种遍历方法(总结)

用java实现的数组创建二叉树以及递归先序遍历,递归中序遍历,递归后序遍历,非递归前序遍历,非递归中序遍历,非递归后序遍历,深度优先遍历,广度优先遍历8种遍历方式:

package myTest;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Stack;

public class myClass {
 
 public static void main(String[] args) {
 // TODO Auto-generated method stub
 myClass tree = new myClass();
 int[] datas = new int[]{1,2,3,4,5,6,7,8,9};
 List<Node> nodelist = new LinkedList<>();
 tree.creatBinaryTree(datas, nodelist);
 Node root = nodelist.get(0);
 System.out.println("递归先序遍历:");
 tree.preOrderTraversal(root);
 System.out.println();
 System.out.println("非递归先序遍历:");
 tree.preOrderTraversalbyLoop(root);
 System.out.println();
 System.out.println("递归中序遍历:");
 tree.inOrderTraversal(root);
 System.out.println();
 System.out.println("非递归中序遍历:");
 tree.inOrderTraversalbyLoop(root);
 System.out.println();
 System.out.println("递归后序遍历:");
 tree.postOrderTraversal(root);
 System.out.println();
 System.out.println("非递归后序遍历:");
 tree.postOrderTraversalbyLoop(root);
 System.out.println();
 System.out.println("广度优先遍历:");
 tree.bfs(root);
 System.out.println();
 System.out.println("深度优先遍历:");
 List<List<Integer>> rst = new ArrayList<>();
 List<Integer> list = new ArrayList<>();
 tree.dfs(root,rst,list);
 System.out.println(rst);
 }
 /**
 * 
 * @param datas 实现二叉树各节点值的数组
 * @param nodelist 二叉树list
 */
 private void creatBinaryTree(int[] datas,List<Node> nodelist){
 //将数组变成node节点
 for(int nodeindex=0;nodeindex<datas.length;nodeindex++){
  Node node = new Node(datas[nodeindex]);
  nodelist.add(node);
 }
 //给所有父节点设定子节点
 for(int index=0;index<nodelist.size()/2-1;index++){
  //编号为n的节点他的左子节点编号为2*n 右子节点编号为2*n+1 但是因为list从0开始编号,所以还要+1
  //这里父节点有1(2,3),2(4,5),3(6,7),4(8,9) 但是最后一个父节点有可能没有右子节点 需要单独处理
  nodelist.get(index).setLeft(nodelist.get(index*2+1)); 
  nodelist.get(index).setRight(nodelist.get(index*2+2));
 }
 //单独处理最后一个父节点 因为它有可能没有右子节点
 int index = nodelist.size()/2-1;
 nodelist.get(index).setLeft(nodelist.get(index*2+1)); //先设置左子节点
 if(nodelist.size() % 2 == 1){ //如果有奇数个节点,最后一个父节点才有右子节点
  nodelist.get(index).setRight(nodelist.get(index*2+2));
 }
 }
 /**
 * 遍历当前节点的值
 * @param nodelist
 * @param node
 */
 public void checkCurrentNode(Node node){
 System.out.print(node.getVar()+" ");
 }
 /**
 * 先序遍历二叉树
 * @param root 二叉树根节点
 */
 public void preOrderTraversal(Node node){
 if (node == null) //很重要,必须加上 当遇到叶子节点用来停止向下遍历
      return; 
 checkCurrentNode(node);
 preOrderTraversal(node.getLeft());
 preOrderTraversal(node.getRight());
 }
 /**
 * 中序遍历二叉树
 * @param root 根节点
 */
 public void inOrderTraversal(Node node){
 if (node == null) //很重要,必须加上
      return; 
 inOrderTraversal(node.getLeft());
 checkCurrentNode(node);
 inOrderTraversal(node.getRight());
 }
 /**
 * 后序遍历二叉树
 * @param root 根节点
 */
 public void postOrderTraversal(Node node){
 if (node == null) //很重要,必须加上
      return; 
 postOrderTraversal(node.getLeft());
 postOrderTraversal(node.getRight());
 checkCurrentNode(node);
 }
 
 /**
 * 非递归前序遍历
 * @param node
 */
 public void preOrderTraversalbyLoop(Node node){
 Stack<Node> stack = new Stack();
 Node p = node;
 while(p!=null || !stack.isEmpty()){
  while(p!=null){ //当p不为空时,就读取p的值,并不断更新p为其左子节点,即不断读取左子节点
  checkCurrentNode(p);
  stack.push(p); //将p入栈
  p = p.getLeft();
  }
  if(!stack.isEmpty()){
  p = stack.pop();
  p = p.getRight();
  }
 }
 }
 /**
 * 非递归中序遍历
 * @param node
 */
 public void inOrderTraversalbyLoop(Node node){
 Stack<Node> stack = new Stack();
 Node p = node;
 while(p!=null || !stack.isEmpty()){
  while(p!=null){
  stack.push(p);
  p = p.getLeft();
  }
  if(!stack.isEmpty()){ 
  p = stack.pop();
  checkCurrentNode(p);
  p = p.getRight();
  }
 }
 }
 /**
 * 非递归后序遍历
 * @param node
 */
 public void postOrderTraversalbyLoop(Node node){
 Stack<Node> stack = new Stack<>();
 Node p = node,prev = node;
 while(p!=null || !stack.isEmpty()){
  while(p!=null){
  stack.push(p);
  p = p.getLeft();
  }
  if(!stack.isEmpty()){
  Node temp = stack.peek().getRight();
  if(temp == null||temp == prev){
   p = stack.pop();
   checkCurrentNode(p);
   prev = p;
   p = null;
  }else{
   p = temp;
  } 
  }
 }
 }
 
 /**
 * 广度优先遍历(从上到下遍历二叉树)
 * @param root
 */
 public void bfs(Node root){
  if(root == null) return;
  LinkedList<Node> queue = new LinkedList<Node>();
  queue.offer(root); //首先将根节点存入队列
  //当队列里有值时,每次取出队首的node打印,打印之后判断node是否有子节点,若有,则将子节点加入队列
  while(queue.size() > 0){ 
  Node node = queue.peek();
   queue.poll(); //取出队首元素并打印
   System.out.print(node.var+" ");
   if(node.left != null){ //如果有左子节点,则将其存入队列
    queue.offer(node.left);
   }
   if(node.right != null){ //如果有右子节点,则将其存入队列
    queue.offer(node.right);
   }
  }
 }
 /**
 * 深度优先遍历
 * @param node
 * @param rst
 * @param list
 */
 public void dfs(Node node,List<List<Integer>> rst,List<Integer> list){
 if(node == null) return;
 if(node.left == null && node.right == null){
  list.add(node.var);
  /* 这里将list存入rst中时,不能直接将list存入,而是通过新建一个list来实现,
  * 因为如果直接用list的话,后面remove的时候也会将其最后一个存的节点删掉*/
  rst.add(new ArrayList<>(list));
  list.remove(list.size()-1);
 }
 list.add(node.var);
 dfs(node.left,rst,list);
 dfs(node.right,rst,list);
 list.remove(list.size()-1);
 }
 /**
 * 节点类
 * var 节点值
 * left 节点左子节点
 * right 右子节点
 */ 
 class Node{
 int var;
 Node left;
 Node right;
 public Node(int var){
  this.var = var;
  this.left = null;
  this.right = null;
 }
 public void setLeft(Node left) {
  this.left = left;
 }
 public void setRight(Node right) {
  this.right = right;
 }
 public int getVar() {
  return var;
 }
 public void setVar(int var) {
  this.var = var;
 }
 public Node getLeft() {
  return left;
 }
 public Node getRight() {
  return right;
 }
 
 }

}

运行结果:

递归先序遍历:
1 2 4 8 9 5 3 6 7

非递归先序遍历:
1 2 4 8 9 5 3 6 7

递归中序遍历:
8 4 9 2 5 1 6 3 7

非递归中序遍历:
8 4 9 2 5 1 6 3 7

递归后序遍历:
8 9 4 5 2 6 7 3 1

非递归后序遍历:
8 9 4 5 2 6 7 3 1

广度优先遍历:
1 2 3 4 5 6 7 8 9

深度优先遍历:
[[1, 2, 4, 8], [1, 2, 4, 9], [1, 2, 5], [1, 3, 6], [1, 3, 7]]

以上这篇java实现二叉树的创建及5种遍历方法(总结)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。


# 二叉树的创建和遍历  # Java递归遍历树形结构的实现代码  # Java遍历输出指定目录、树形结构所有文件包括子目录下的文件  # 图解二叉树的三种遍历方式及java实现代码  # 图解红黑树及Java进行红黑二叉树遍历的方法  # Java实现二叉树的深度优先遍历和广度优先遍历算法示例  # java实现遍历树形菜单两种实现代码分享  # Java二叉树的四种遍历方式详解  # 简单谈谈Java遍历树深度优先和广度优先的操作方式  # 遍历  # 递归  # 二叉树  # 很重要  # 则将  # 有可能  # 给大家  # 也会  # 将其  # 希望能  # 才有  # 为其  # 因为它  # 若有  # 这篇  # 来实现  # 小编  # 数个  # 不断更新  # 大家多多 


相关文章: 婚礼视频制作网站,学习*后期制作的网站有哪些?  公司门户网站制作流程,华为官网怎么做?  Python lxml的etree和ElementTree有什么区别  云南网站制作公司有哪些,云南最好的招聘网站是哪个?  如何在橙子建站中快速调整背景颜色?  企业宣传片制作网站有哪些,传媒公司怎么找企业宣传片项目?  昆明网站制作哪家好,昆明公租房申请网上登录入口?  ,制作一个手机app网站要多少钱?  如何处理“XML格式不正确”错误 常见XML well-formed问题解决方法  家具网站制作软件,家具厂怎么跑业务?  网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?  网站制作公司,橙子建站是合法的吗?  深圳企业网站制作设计,在深圳如何网上全流程注册公司?  营销式网站制作方案,销售哪个网站招聘效果最好?  免费公司网站制作软件,如何申请免费主页空间做自己的网站?  建站之星后台管理系统如何操作?  怎么将XML数据可视化 D3.js加载XML  如何通过NAT技术实现内网高效建站?  网站设计制作公司地址,网站建设比较好的公司都有哪些?  c# F# 的 MailboxProcessor 和 C# 的 Actor 模型  制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?  移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?  如何在建站主机中优化服务器配置?  php8.4新语法match怎么用_php8.4match表达式替代switch【方法】  建站之星上传入口如何快速找到?  如何通过云梦建站系统实现SEO快速优化?  儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?  如何快速搭建二级域名独立网站?  济南网站建设制作公司,室内设计网站一般都有哪些功能?  建站之星免费版是否永久可用?  如何在搬瓦工VPS快速搭建网站?  北京的网站制作公司有哪些,哪个视频网站最好?  音乐网站服务器如何优化API响应速度?  深圳 网站制作,深圳招聘网站哪个比较好一点啊?  制作网站的公司有哪些,做一个公司网站要多少钱?  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  海南网站制作公司有哪些,海口网是哪家的?  建站之星如何保障用户数据免受黑客入侵?  郑州企业网站制作公司,郑州招聘网站有哪些?  如何选择域名并搭建高效网站?  网站制作壁纸教程视频,电脑壁纸网站?  建站之星安装后如何自定义网站颜色与字体?  武汉网站如何制作,黄黄高铁武穴北站途经哪些村庄?  建站之星如何防范黑客攻击与数据泄露?  如何正确下载安装西数主机建站助手?  如何选择长沙网站建站模板?H5响应式与品牌定制哪个更优?  宝华建站服务条款解析:五站合一功能与SEO优化设置指南  专业网站制作企业网站,如何制作一个企业网站,建设网站的基本步骤有哪些?  建站之星如何实现网站加密操作?  中山网站制作网页,中山新生登记系统登记流程? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。