韩顺平 数据结构与算法 (11_1) 树结构基础部分_二叉树
3)二叉树
1. 二叉树的概念
树有很多种,每个节点最多只有两个子节点的形式就是二叉树
二叉树的子节点分为左节点和右节点
如果该二叉树的所有叶子节点都在最后一层并且结点总数=2^n-1(n为层数)我们称为满二叉树
如果该二叉树的所有叶子节点都在最后一层或者倒数第二层,而且最后一层的叶子节点在左边连续,倒数第二层的叶子节点在右边连续,我们称之为完全二叉树
PS:连续是横向连续不是纵向连续(比如81->91左连续;71->61->15右连续)
2. 二叉树遍历说明
前序遍历:先输出父节点,再遍历左子树和右子树
中序遍历:先遍历左子树,再输出父节点,再遍历右子树
后续遍历:先遍历左子树,再遍历右子树,最后输出父节点
小结:看输出父节点的顺序,就确定是前序,中序还是后序
思路分析:
- 创建一颗二叉树
- 前序遍历
- 先输出当前节点(初始的时候是root节点);
- 如果左子节点不为空,则递归前序遍历;
- 如果右子节点不为空,则递归前序遍历
- 中序遍历
- 如果当前节点的左子节点不为空,则递归中序遍历;
- 输出当前节点
- 如果当前节点的右子节点不为空,则递归中序遍历
- 后序遍历
- 如果当前节点的左子节点不为空,则递归后序遍历
- 如果当前节点的右子节点不为空,则递归后序遍历
- 输出当前节点
代码实现(遍历部分)
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. 二叉树的检索(查询)
要求:
- 请编写前中后序查找方法
- 分别用三种方法查找hero.no==5的节点
- 分析三种查找方法,进行比较
思路:
前序查找
- 先判断当前节点的no是否等于要查找的
- 相等:则返回当前节点
- 不等,则判断左子节点是否为空,若不空则递归前序查找,一旦找到就返回;
- 左边前序递归完未找到,再判断右节点是否为空,不为空就前序递归右边,一旦找到就返回;
中序查找
- 判断当前节点的左子节点是否为空,不为空则前序递归查找
- 如果找到就返回,否则就和当前节点比较,如果是就返回当前节点,否则继续进行右递归的中序查找
- 如果右递归中序查找找到就返回,否则返回null
后序查找
- 先判断当前节点的左子节点是否为空,不为空则递归后序遍历
- 如果找到,就返回,如果没有找到,就判断当前节点的右子节点是否为空,如果不为空,则右递归进行后序查找,没会找到就返回
- 最后和当前接待你进行比较,如果是就返回,不是就返回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. 二叉树的删除
要求:
- 如果删除的节点是叶子节点,则删除该节点
- 如果删除的节点是非叶子节点,则删除改子树(后序再说树叶上提的问题)
- 测试,删除掉5号叶子节点和3号子树
思路:
- 如果只有一个root节点,就等价将二叉树置空
- 因为二叉树是单向的,所以我们是判断当前节点的子节点是否需要删除;而不能判断当前节点是否需要删除
- 如果当前节点的左子节点不为空,并且左子节点就是要删除的节点,就将this.left = null;并且就返回(结束递归删除)
- 如果当前节点的右子节点不为空,并且右子节点就是要删除的节点,就将this.right = null;并且就返回(结束递归删除)
- 如果第2步和第3步都没有删除节点,那么我们就需要向左子树进行递归删除;
- 如果第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) 树结构基础部分_二叉树相关推荐
- 尚硅谷Java数据结构和java算法,韩顺平数据结构和算法课后作业01
尚硅谷Java数据结构和java算法,韩顺平数据结构和算法课后作业第一题 要求: 1)在前面的基础上,将稀疏数组保存到磁盘上,比如map.data 2) 恢复原来的数组时,读取map.data进行恢复 ...
- 韩顺平 数据结构与算法 (12_6) 树结构应用部分_赫夫曼编码实践——数据解码
赫夫曼编码实践--数据解码 代码实现(承袭压缩的代码) package DataStructures.huffmanTree;import javax.print.DocFlavor; import ...
- 韩顺平 数据结构与算法 (12_3) 树结构应用部分_赫夫曼编码(思路)
赫夫曼编码(思路) 1.基本介绍 赫夫曼编码也翻译为哈夫曼编码(Huffman Coding),又称赫夫曼编码, 赫夫曼编码是赫哈夫曼树在电讯通信中的经典的应用之一 赫夫曼编码广泛的用于数据文件压缩与 ...
- 数据结构与算法之树结构基础
目录 为什么要使用树结构 树结构基本概念 树的种类 树的存储与表示 常见的一些树的应用场景 为什么要使用树结构 线性结构中不论是数组还是链表,他们都存在着诟病:比如查找某个数必须从头开始查,消耗较多的 ...
- 韩顺平 数据结构与算法 (12_4) 赫夫曼编码(代码)
实际案例--数据压缩 目标: 将给出的一段文本,比如"ilikelikelikejavadoyoulikeajava",根据前面的的赫夫曼编码原理,对其进行数据压缩处理步骤1:根据 ...
- Interview之AI:人工智能领域岗位求职面试—人工智能算法工程师知识框架及课程大纲(AI基础之数学基础/数据结构与算法/编程学习基础、ML算法简介、DL算法简介)来理解技术交互流程
Interview之AI:人工智能领域岗位求职面试-人工智能算法工程师知识框架及课程大纲(AI基础之数学基础/数据结构与算法/编程学习基础.ML算法简介.DL算法简介)来理解技术交互流程 目录 一.A ...
- 『数据结构与算法』解读树(Tree)和二叉树(Binary Tree)!
『数据结构与算法』解读树(Tree)和二叉树(Binary Tree)! 文章目录 一. 树 1.1. 树的定义 1.2. 树的基本术语 1.3. 树的性质 二. 二叉树 2.1. 二叉树的定义 2. ...
- 【超全汇总】学习数据结构与算法,计算机基础知识,看这篇就够了【ZT帅地】2020-3-7
https://blog.csdn.net/m0_37907797/article/details/104029002 由于文章有点多,并且发的文章也不是一个系列一个系列发的,不过我的文章大部分都是围 ...
- mooc数据结构与算法python版期末测验_中国大学MOOC(慕课)_数据结构与算法Python版_测试题及答案...
中国大学MOOC(慕课)_数据结构与算法Python版_测试题及答案 更多相关问题 采用fopen()函数打开文件,支持文件读取的参数有: [简答题]简单阐述高分子材料热-机械特征及成型加工的关系,并 ...
最新文章
- R语言使用ggplot2可视化凹凸图(bumps chart、凹凸图是一种特殊形式的线图,旨在探索随着时间的推移等级的变化)、并设置凹凸图的线条为曲线而不是直线(change into curves)
- zabbix如何配置微信报警
- 批量操作WinRAR实用技巧七招
- [精讲-3]Offline Domain Join
- 《linux操作系统》第06章在线测试,Linux系统管理一测试题-附答案.doc
- php的全局p变量程序_php全局变量的使用
- bzoj 1579: [Usaco2009 Feb]Revamping Trails 道路升级【分层图+spfa】
- 剑指offer面试题:输入某二叉树的前序遍历和中序遍历,输出后序遍历
- 记一次spirngMVC整合HttpPrinter的过程
- ubuntu风扇转速控制与系统状态监控
- input type=file 的onchange事件
- 从零开始学《离散数学》-电子科技大学
- python 拆分excel单元格_python使用openpyxl excel 合并拆分单元格
- 第12章实验1:学生成绩管理系统V5.0
- zoom 用法 flex zoom
- Nero8直接把APE带CUE映像文件刻录CD方法(转帖)
- 在Matlab中将一幅图片的中心设置为x-y轴坐标原点
- 欧拉函数求互质数的个数
- 【半导体先进工艺制程技术系列】SOI技术(下)
- Testing Process - 读书笔记