3)二叉树
1. 二叉树的概念
  1. 树有很多种,每个节点最多只有两个子节点的形式就是二叉树

  2. 二叉树的子节点分为左节点和右节点

  3. 如果该二叉树的所有叶子节点都在最后一层并且结点总数=2^n-1(n为层数)我们称为满二叉树

  4. 如果该二叉树的所有叶子节点都在最后一层或者倒数第二层,而且最后一层的叶子节点在左边连续,倒数第二层的叶子节点在右边连续,我们称之为完全二叉树
    PS:连续是横向连续不是纵向连续(比如81->91左连续;71->61->15右连续)

2. 二叉树遍历说明

前序遍历:先输出父节点,再遍历左子树和右子树

中序遍历:先遍历左子树,再输出父节点,再遍历右子树

后续遍历:先遍历左子树,再遍历右子树,最后输出父节点

小结:看输出父节点的顺序,就确定是前序,中序还是后序

思路分析:

  1. 创建一颗二叉树
  2. 前序遍历
    1. 先输出当前节点(初始的时候是root节点);
    2. 如果左子节点不为空,则递归前序遍历;
    3. 如果右子节点不为空,则递归前序遍历
  3. 中序遍历
    1. 如果当前节点的左子节点不为空,则递归中序遍历;
    2. 输出当前节点
    3. 如果当前节点的右子节点不为空,则递归中序遍历
  4. 后序遍历
    1. 如果当前节点的左子节点不为空,则递归后序遍历
    2. 如果当前节点的右子节点不为空,则递归后序遍历
    3. 输出当前节点

代码实现(遍历部分)

MainBinaryTree binaryTree = new BinaryTree();//创先需要的节点HeroNode root = new HeroNode(1, "宋江");HeroNode node2 = new HeroNode(2, "吴用");HeroNode node3 = new HeroNode(3, "卢俊义");HeroNode node4 = new HeroNode(4, "林冲");HeroNode node5 = new HeroNode(5, "关胜");//说明:我们先手动创建二叉树,后面使用递归创建二叉树root.setLeft(node2);root.setRight(node3);node3.setRight(node4);node3.setLeft(node5);binaryTree.setRoot(root);BinaryTree>Search//前序遍历public void preOrder() {if (this.root != null) {this.root.preOrder();} else {System.out.println("二叉树为空,无法遍历");}}//中序遍历public void infixOrder() {if (this.root != null) {this.root.infixOrder();} else {System.out.println("二叉树为空,无法遍历");}}//后序遍历public void postOrder() {if (this.root != null) {this.root.postOrder();} else {System.out.println("二叉树为空,无法遍历");}}HeroNode>Search
// 1. 前序遍历public void preOrder() {//先输出父节点System.out.println(this);//向左子树前序遍历递归if (this.left != null) {this.left.preOrder();}//向右子树前序遍历递归if (this.right != null) {this.right.preOrder();}}// 2. 中序遍历public void infixOrder() {//向左子树中序遍历递归if (this.left != null) {this.left.infixOrder();}System.out.println(this);//向右子树中序遍历递归if (this.right != null) {this.right.infixOrder();}}// 3. 后序遍历public void postOrder() {//向左子树后序遍历递归iif (this.left != null) {this.left.postOrder();}if (this.right != null) {this.right.postOrder();}System.out.println(this);}
3. 二叉树的检索(查询)

要求:

  1. 请编写前中后序查找方法
  2. 分别用三种方法查找hero.no==5的节点
  3. 分析三种查找方法,进行比较

思路:

  1. 前序查找

    1. 先判断当前节点的no是否等于要查找的
    2. 相等:则返回当前节点
    3. 不等,则判断左子节点是否为空,若不空则递归前序查找,一旦找到就返回;
    4. 左边前序递归完未找到,再判断右节点是否为空,不为空就前序递归右边,一旦找到就返回;
  2. 中序查找

    1. 判断当前节点的左子节点是否为空,不为空则前序递归查找
    2. 如果找到就返回,否则就和当前节点比较,如果是就返回当前节点,否则继续进行右递归的中序查找
    3. 如果右递归中序查找找到就返回,否则返回null
  3. 后序查找

    1. 先判断当前节点的左子节点是否为空,不为空则递归后序遍历
    2. 如果找到,就返回,如果没有找到,就判断当前节点的右子节点是否为空,如果不为空,则右递归进行后序查找,没会找到就返回
    3. 最后和当前接待你进行比较,如果是就返回,不是就返回null

代码实现(检索部分)

Main//前序遍历查找System.out.println("前序遍历方式~~~~");HeroNode resNode_pre = binaryTree.preOrderSearch(5);if (resNode_pre != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_pre.getNo(), resNode_pre.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}//中序遍历查找——3次System.out.println("中序遍历方式!!!!");HeroNode resNode_infix = binaryTree.infixOrderSearch(5);if (resNode_infix != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_infix.getNo(), resNode_infix.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}//后序遍历查找——2次System.out.println("后序遍历方式····");HeroNode resNode_Post = binaryTree.postOrderSearch(5);if (resNode_Post != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_Post.getNo(), resNode_Post.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}BinaryTree>preOrderSearch//前序查找public HeroNode preOrderSearch(int no) {if (root != null) {return root.preOrderSearch(no);} else {return null;}}
BinaryTree>infixOrderSearch
//中序查找public HeroNode infixOrderSearch(int no) {if (root != null) {return root.infixOrderSearch(no);} else {return null;}}
BinaryTree>postOrderSearch//后序查找public HeroNode postOrderSearch(int no) {if (root != null) {return root.postOrderSearch(no);} else {return null;}}HeroNode>preOrderSearch// 1. 前序遍历查找public HeroNode preOrderSearch(int no) {System.out.println("进入前序遍历查找~~~");//自己if (this.no == no) {return this;}//向左HeroNode resNode = null;if (this.left != null) {resNode = this.left.preOrderSearch(no);}if (resNode != null) {return resNode;}//向右if (this.right != null) {resNode = this.right.preOrderSearch(no);}return resNode;}
HeroNode>infixOrderSearch// 2. 中序遍历查找public HeroNode infixOrderSearch(int no) {//向左HeroNode resNode = null;if (this.left != null) {resNode = this.left.infixOrderSearch(no);}if (resNode != null) {return resNode;}//自己System.out.println("进入中序查找");if (this.no == no) {return this;}//向右if (this.right != null) {resNode = this.right.infixOrderSearch(no);}return resNode;}
HeroNode>postOrderSearch// 3. 后序遍历查找public HeroNode postOrderSearch(int no) {HeroNode resNode = null;//向左if (this.left != null) {resNode = this.left.postOrderSearch(no);}if (resNode != null) {return resNode;}//向右if (this.right != null) {resNode = this.right.postOrderSearch(no);}if (resNode != null) {return resNode;}//自己System.out.println("后序遍历");if (this.no == no) {return this;}return resNode;}
4. 二叉树的删除

要求:

  1. 如果删除的节点是叶子节点,则删除该节点
  2. 如果删除的节点是非叶子节点,则删除改子树(后序再说树叶上提的问题)
  3. 测试,删除掉5号叶子节点和3号子树

思路:

  1. 如果只有一个root节点,就等价将二叉树置空
  2. 因为二叉树是单向的,所以我们是判断当前节点的子节点是否需要删除;而不能判断当前节点是否需要删除
  3. 如果当前节点的左子节点不为空,并且左子节点就是要删除的节点,就将this.left = null;并且就返回(结束递归删除)
  4. 如果当前节点的右子节点不为空,并且右子节点就是要删除的节点,就将this.right = null;并且就返回(结束递归删除)
  5. 如果第2步和第3步都没有删除节点,那么我们就需要向左子树进行递归删除;
  6. 如果第4步也没有删除节点,那么我们就需要向右子树进行递归删除;

代码实现(删除部分)

Main:
//测试删除System.out.println("删除前,前序遍历");binaryTree.preOrder();//1->2->3->5->4//binaryTree.delNode(5);//删除子叶binaryTree.delNode(3);//删除子树System.out.println("删除后,前序遍历");//binaryTree.preOrder();//1->2->3->4binaryTree.preOrder();//1->2
BinaryTree>delNode
//删除节点public void delNode(int no){if (root!=null){//需要判断root是否是要删除的节点if (root.getNo()==no){root=null;}else {//递归删除root.delNode(no);}}else {System.out.println("空树,不能删除");}}
HeroNode>delNode
//递归删除节点//1.如果删除叶子节点,就直接删除//2.如果删除非叶子节点,就删除该子树public void delNode(int no) {//判断该节点的左右是否需要删除if (this.left != null && this.left.no == no) {this.left = null;return;}if (this.right != null && this.right.no == no) {this.right = null;return;}//上两步没完成任务,开始递归寻找if (this.left != null) {this.left.delNode(no);}if (this.right != null) {this.right.delNode(no);}}
代码实现【大汇总】——遍历+查询+删除(简单删除)
package DataStructures.Tree;public class BinaryTreeDemo {public static void main(String[] args) {//先创建一颗二叉树BinaryTree binaryTree = new BinaryTree();//创先需要的节点HeroNode root = new HeroNode(1, "宋江");HeroNode node2 = new HeroNode(2, "吴用");HeroNode node3 = new HeroNode(3, "卢俊义");HeroNode node4 = new HeroNode(4, "林冲");HeroNode node5 = new HeroNode(5, "关胜");//说明:我们先手动创建二叉树,后面使用递归创建二叉树root.setLeft(node2);root.setRight(node3);node3.setRight(node4);node3.setLeft(node5);binaryTree.setRoot(root);
/*        //测试:System.out.println("前序遍历");//1->2->3->5->4binaryTree.preOrder();//测试:System.out.println("中序遍历");//2->1->5->3->4binaryTree.infixOrder();//测试:System.out.println("后序遍历");//2->5->4->3->1binaryTree.postOrder();
*/
/*      //前序遍历查找System.out.println("前序遍历方式~~~~");HeroNode resNode_pre = binaryTree.preOrderSearch(5);if (resNode_pre != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_pre.getNo(), resNode_pre.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}//中序遍历查找——3次System.out.println("中序遍历方式!!!!");HeroNode resNode_infix = binaryTree.infixOrderSearch(5);if (resNode_infix != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_infix.getNo(), resNode_infix.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}//后序遍历查找——2次System.out.println("后序遍历方式····");HeroNode resNode_Post = binaryTree.postOrderSearch(5);if (resNode_Post != null) {System.out.printf("找到了,信息为no=%d name=%s\n", resNode_Post.getNo(), resNode_Post.getName());} else {System.out.printf("没有找到no = %d的英雄\n", 5);}*///测试删除System.out.println("删除前,前序遍历");binaryTree.preOrder();//1->2->3->5->4//binaryTree.delNode(5);//删除子叶binaryTree.delNode(3);//删除子树System.out.println("删除后,前序遍历");//binaryTree.preOrder();//1->2->3->4binaryTree.preOrder();//1->2}
}//定义BinaryTree二叉树
class BinaryTree {private HeroNode root;public void setRoot(HeroNode root) {this.root = root;}//删除节点public void delNode(int no){if (root!=null){//需要判断root是否是要删除的节点if (root.getNo()==no){root=null;}else {//递归删除root.delNode(no);}}else {System.out.println("空树,不能删除");}}//前序遍历public void preOrder() {if (this.root != null) {this.root.preOrder();} else {System.out.println("二叉树为空,无法遍历");}}//中序遍历public void infixOrder() {if (this.root != null) {this.root.infixOrder();} else {System.out.println("二叉树为空,无法遍历");}}//后序遍历public void postOrder() {if (this.root != null) {this.root.postOrder();} else {System.out.println("二叉树为空,无法遍历");}}//前序查找public HeroNode preOrderSearch(int no) {if (root != null) {return root.preOrderSearch(no);} else {return null;}}//中序查找public HeroNode infixOrderSearch(int no) {if (root != null) {return root.infixOrderSearch(no);} else {return null;}}//后序查找public HeroNode postOrderSearch(int no) {if (root != null) {return root.postOrderSearch(no);} else {return null;}}
}//先创建HeroNode结点
class HeroNode {private int no;private String name;private HeroNode left;private HeroNode right;public HeroNode(int no, String name) {this.no = no;this.name = name;}public int getNo() {return no;}public void setNo(int no) {this.no = no;}public String getName() {return name;}public void setName(String name) {this.name = name;}public HeroNode getLeft() {return left;}public void setLeft(HeroNode left) {this.left = left;}public HeroNode getRight() {return right;}public void setRight(HeroNode right) {this.right = right;}@Overridepublic String toString() {return "HeroNode[no=" + no + ", name=" + name + "]";}//递归删除节点//1.如果删除叶子节点,就直接删除//2.如果删除非叶子节点,就删除该子树public void delNode(int no) {//判断该节点的左右是否需要删除if (this.left != null && this.left.no == no) {this.left = null;return;}if (this.right != null && this.right.no == no) {this.right = null;return;}//上两步没完成任务,开始递归寻找if (this.left != null) {this.left.delNode(no);}if (this.right != null) {this.right.delNode(no);}}// 1. 前序遍历public void preOrder() {//先输出父节点System.out.println(this);//向左子树前序遍历递归if (this.left != null) {this.left.preOrder();}//向右子树前序遍历递归if (this.right != null) {this.right.preOrder();}}// 2. 中序遍历public void infixOrder() {//向左子树中序遍历递归if (this.left != null) {this.left.infixOrder();}System.out.println(this);//向右子树中序遍历递归if (this.right != null) {this.right.infixOrder();}}// 3. 后序遍历public void postOrder() {//向左子树后序遍历递归iif (this.left != null) {this.left.postOrder();}if (this.right != null) {this.right.postOrder();}System.out.println(this);}// 1. 前序遍历查找/*** @param no 查找no* @return 如果找到就返回该Node,如果没有找到就返回null*/public HeroNode preOrderSearch(int no) {System.out.println("进入前序遍历查找~~~");//自己if (this.no == no) {return this;}//向左HeroNode resNode = null;if (this.left != null) {resNode = this.left.preOrderSearch(no);}if (resNode != null) {return resNode;}//向右if (this.right != null) {resNode = this.right.preOrderSearch(no);}return resNode;}// 2. 中序遍历查找public HeroNode infixOrderSearch(int no) {//向左HeroNode resNode = null;if (this.left != null) {resNode = this.left.infixOrderSearch(no);}if (resNode != null) {return resNode;}//自己System.out.println("进入中序查找");if (this.no == no) {return this;}//向右if (this.right != null) {resNode = this.right.infixOrderSearch(no);}return resNode;}// 3. 后序遍历查找public HeroNode postOrderSearch(int no) {HeroNode resNode = null;//向左if (this.left != null) {resNode = this.left.postOrderSearch(no);}if (resNode != null) {return resNode;}//向右if (this.right != null) {resNode = this.right.postOrderSearch(no);}if (resNode != null) {return resNode;}//自己System.out.println("后序遍历");if (this.no == no) {return this;}return resNode;}
}

韩顺平 数据结构与算法 (11_1) 树结构基础部分_二叉树相关推荐

  1. 尚硅谷Java数据结构和java算法,韩顺平数据结构和算法课后作业01

    尚硅谷Java数据结构和java算法,韩顺平数据结构和算法课后作业第一题 要求: 1)在前面的基础上,将稀疏数组保存到磁盘上,比如map.data 2) 恢复原来的数组时,读取map.data进行恢复 ...

  2. 韩顺平 数据结构与算法 (12_6) 树结构应用部分_赫夫曼编码实践——数据解码

    赫夫曼编码实践--数据解码 代码实现(承袭压缩的代码) package DataStructures.huffmanTree;import javax.print.DocFlavor; import ...

  3. 韩顺平 数据结构与算法 (12_3) 树结构应用部分_赫夫曼编码(思路)

    赫夫曼编码(思路) 1.基本介绍 赫夫曼编码也翻译为哈夫曼编码(Huffman Coding),又称赫夫曼编码, 赫夫曼编码是赫哈夫曼树在电讯通信中的经典的应用之一 赫夫曼编码广泛的用于数据文件压缩与 ...

  4. 数据结构与算法之树结构基础

    目录 为什么要使用树结构 树结构基本概念 树的种类 树的存储与表示 常见的一些树的应用场景 为什么要使用树结构 线性结构中不论是数组还是链表,他们都存在着诟病:比如查找某个数必须从头开始查,消耗较多的 ...

  5. 韩顺平 数据结构与算法 (12_4) 赫夫曼编码(代码)

    实际案例--数据压缩 目标: 将给出的一段文本,比如"ilikelikelikejavadoyoulikeajava",根据前面的的赫夫曼编码原理,对其进行数据压缩处理步骤1:根据 ...

  6. Interview之AI:人工智能领域岗位求职面试—人工智能算法工程师知识框架及课程大纲(AI基础之数学基础/数据结构与算法/编程学习基础、ML算法简介、DL算法简介)来理解技术交互流程

    Interview之AI:人工智能领域岗位求职面试-人工智能算法工程师知识框架及课程大纲(AI基础之数学基础/数据结构与算法/编程学习基础.ML算法简介.DL算法简介)来理解技术交互流程 目录 一.A ...

  7. 『数据结构与算法』解读树(Tree)和二叉树(Binary Tree)!

    『数据结构与算法』解读树(Tree)和二叉树(Binary Tree)! 文章目录 一. 树 1.1. 树的定义 1.2. 树的基本术语 1.3. 树的性质 二. 二叉树 2.1. 二叉树的定义 2. ...

  8. 【超全汇总】学习数据结构与算法,计算机基础知识,看这篇就够了【ZT帅地】2020-3-7

    https://blog.csdn.net/m0_37907797/article/details/104029002 由于文章有点多,并且发的文章也不是一个系列一个系列发的,不过我的文章大部分都是围 ...

  9. mooc数据结构与算法python版期末测验_中国大学MOOC(慕课)_数据结构与算法Python版_测试题及答案...

    中国大学MOOC(慕课)_数据结构与算法Python版_测试题及答案 更多相关问题 采用fopen()函数打开文件,支持文件读取的参数有: [简答题]简单阐述高分子材料热-机械特征及成型加工的关系,并 ...

最新文章

  1. R语言使用ggplot2可视化凹凸图(bumps chart、凹凸图是一种特殊形式的线图,旨在探索随着时间的推移等级的变化)、并设置凹凸图的线条为曲线而不是直线(change into curves)
  2. zabbix如何配置微信报警
  3. 批量操作WinRAR实用技巧七招
  4. [精讲-3]Offline Domain Join
  5. 《linux操作系统》第06章在线测试,Linux系统管理一测试题-附答案.doc
  6. php的全局p变量程序_php全局变量的使用
  7. bzoj 1579: [Usaco2009 Feb]Revamping Trails 道路升级【分层图+spfa】
  8. 剑指offer面试题:输入某二叉树的前序遍历和中序遍历,输出后序遍历
  9. 记一次spirngMVC整合HttpPrinter的过程
  10. ubuntu风扇转速控制与系统状态监控
  11. input type=file 的onchange事件
  12. 从零开始学《离散数学》-电子科技大学
  13. python 拆分excel单元格_python使用openpyxl excel 合并拆分单元格
  14. 第12章实验1:学生成绩管理系统V5.0
  15. zoom 用法 flex zoom
  16. Nero8直接把APE带CUE映像文件刻录CD方法(转帖)
  17. 在Matlab中将一幅图片的中心设置为x-y轴坐标原点
  18. 欧拉函数求互质数的个数
  19. 【半导体先进工艺制程技术系列】SOI技术(下)
  20. Testing Process - 读书笔记

热门文章

  1. 动态规划:国王与金矿
  2. dbconfig设置mysql_dbconfig配置 | 学步园
  3. CSS常用英文字体介绍
  4. oracle查询数据泵状态,PLSQL_数据泵导入进度查看Impdp/Expdp Status(案例)
  5. 最后一次谈开源与免费
  6. ArcGIS10 图框生成和批量打印工具V5.0正式发布
  7. 服务器和交换机直连,网口灯不亮问题排查
  8. 谈下mysql中间件(问题域、业内组件)
  9. 串口接收模块uart_rx详解
  10. Hdu.1248.寒冰王座