一、二叉树

1、为什么需要树这种数据结构(数组、链表、树说明)

1) 数组存储方式的分析(缺点插入)

QQ截图20200826093424.jpg

2) 链式存储方式的分析(缺点检索)

QQ截图20200826093508.jpgQQ截图20200826093515.jpg

3) 树存储方式的分析(提高存储读取)

QQ截图20200826093558.jpg

2、树示意图(常用术语)

QQ截图20200826094521.jpgQQ截图20200826094527.jpg

3、二叉树的概念(包含满二叉树及完全二叉树)

说明:二叉树的介绍以及满二叉树和完全二叉树的介绍

QQ截图20200826094652.jpg

4、二叉树遍历应用实例(前序,中序,后序)

源代码:BinaryTree.java

思路分析

注意的是这里的方法增加于上面的类中进行实现
QQ截图20200826100308.jpg

代码实现(完整过程+测试)

源代码:BinaryTree.java

说明:分为两个类,一个为二叉树节点类包含前中后序遍历,另一个是二叉树类包含根节点以及进行前中后序的调用方法。
**

节点类

  1. //树的节点
  2. class TreeNode{
  3. private String name;
  4. private int id;
  5. private TreeNode left;
  6. private TreeNode right;
  7. public TreeNode(String name, int id) {
  8. super();
  9. this.name = name;
  10. this.id = id;
  11. }
  12. //前序遍历
  13. public void preOrder() {
  14. System.out.println(this);
  15. //如果左节点不为空那么向左递归
  16. if(this.left != null) {
  17. this.left.preOrder();
  18. }
  19. //右节点不为空进行向右递归
  20. if(this.right != null) {
  21. this.right.preOrder();
  22. }
  23. }
  24. //中序遍历
  25. public void infixOrder() {
  26. // 如果左节点不为空那么向左递归
  27. if (this.left != null) {
  28. this.left.infixOrder();
  29. }
  30. System.out.println(this);
  31. // 右节点不为空进行向右递归
  32. if (this.right != null) {
  33. this.right.infixOrder();
  34. }
  35. }
  36. //后序遍历
  37. public void postOrder() {
  38. // 如果左节点不为空那么向左递归
  39. if (this.left != null) {
  40. this.left.postOrder();
  41. }
  42. // 右节点不为空进行向右递归
  43. if (this.right != null) {
  44. this.right.postOrder();
  45. }
  46. System.out.println(this);
  47. }
  48. public void setName(String name) {
  49. this.name = name;
  50. }
  51. public void setId(int id) {
  52. this.id = id;
  53. }
  54. public void setLeft(TreeNode left) {
  55. this.left = left;
  56. }
  57. public void setRight(TreeNode right) {
  58. this.right = right;
  59. }
  60. @Override
  61. public String toString() {
  62. return "TreeNode [name=" + name + ", id=" + id + "]";
  63. }
  64. }

二叉树类

  1. //二叉树
  2. class MyBinaryTree{
  3. //设置一个根节点
  4. private TreeNode root;
  5. public void setRoot(TreeNode root) {
  6. this.root = root;
  7. }
  8. //前序遍历
  9. public void preOrder() {
  10. //判断根节点是否为空
  11. if(root != null) {
  12. root.preOrder();//进行前序遍历
  13. }else {
  14. System.out.println("根节点为空无法前序遍历!");
  15. }
  16. }
  17. // 中序遍历
  18. public void infixOrder() {
  19. if (root != null) {
  20. root.infixOrder();// 进行中序遍历
  21. }else {
  22. System.out.println("根节点为空无法中序遍历!");
  23. }
  24. }
  25. // 后序遍历
  26. public void postOrder() {
  27. if (root != null) {
  28. root.postOrder();// 进行后序遍历
  29. }else {
  30. System.out.println("根节点为空无法后序遍历!");
  31. }
  32. }
  33. }

测试类

  1. package tree;
  2. public class BinaryTree {
  3. public static void main(String[] args) {
  4. //创建一个二叉树
  5. MyBinaryTree tree = new MyBinaryTree();
  6. //创建四个节点
  7. TreeNode root = new TreeNode("宋江", 1);
  8. TreeNode node2 = new TreeNode("吴用", 2);
  9. TreeNode node3 = new TreeNode("卢俊", 3);
  10. TreeNode node4 = new TreeNode("林冲", 4);
  11. //将node节点存放到root中
  12. root.setLeft(node2);
  13. root.setRight(node3);
  14. node3.setRight(node4);
  15. //将root存放到二叉树中
  16. tree.setRoot(root);
  17. //进行前序遍历
  18. System.out.println("前序遍历:");
  19. tree.preOrder();
  20. System.out.println("中序遍历:");
  21. tree.infixOrder();
  22. System.out.println("后序遍历:");
  23. tree.postOrder();
  24. }
  25. }

运行结果:
QQ截图20200826105426.jpg
**

小测试

说说下图的前中后遍历的结果
QQ截图20200826105546.jpg
前序:12354
中序:21534
后序:25431
**

5、查找指定节点(前中后序)

思路分析

与之前前后序遍历几乎相同,只不过这里面增加的返回值以及比较判断的过程。
QQ截图20200827104826.jpg


代码实现

说明:对于节点类以及二叉树类都会有前中后序的方法实现
**

节点类

说明:可以看到每种查找几乎都只是顺序变了而已,其中对于查找到的值使用resultNode来接收并根据实际情况返回,并且在判断比较id时才是真正开始比较,这里输出来体现。

  1. //前序遍历查找
  2. public TreeNode preOrderSearch(int findId) {
  3. System.out.println("前序比较~");
  4. if(this.id == findId) {
  5. return this;
  6. }
  7. //进行向左子树进行查找
  8. TreeNode resultNode = null;
  9. if(this.left != null) {
  10. resultNode = this.left.preOrderSearch(findId);//向左进行查找返回
  11. }
  12. if(resultNode != null) {
  13. return resultNode;
  14. }
  15. //向右子树进行查找
  16. if(this.right != null) {
  17. resultNode = this.right.preOrderSearch(findId);//向右查找返回
  18. }
  19. return resultNode;//返回对应查找到的值
  20. }
  21. //中序遍历查找
  22. public TreeNode infixOrderSearch(int findId) {
  23. // 首先进行向左子树进行查找
  24. TreeNode resultNode = null;
  25. if (this.left != null) {
  26. resultNode = this.left.infixOrderSearch(findId);// 向左进行查找返回
  27. }
  28. if(resultNode != null) {
  29. return resultNode;
  30. }
  31. System.out.println("中序比较~");
  32. if(this.id == findId) {
  33. return this;
  34. }
  35. //向右子树进行查找
  36. if(this.right != null) {
  37. resultNode = this.right.infixOrderSearch(findId);//向右查找返回
  38. }
  39. return resultNode;//返回对应查找到的值
  40. }
  41. //后序遍历查找
  42. public TreeNode postOrderSearch(int findId) {
  43. // 首先进行向左子树进行查找
  44. TreeNode resultNode = null;
  45. if (this.left != null) {
  46. resultNode = this.left.infixOrderSearch(findId);// 向左进行查找返回
  47. }
  48. if(resultNode != null) {
  49. return resultNode;
  50. }
  51. //向右子树进行查找
  52. if (this.right != null) {
  53. resultNode = this.right.infixOrderSearch(findId);// 向右查找返回
  54. }
  55. if(resultNode != null) {
  56. return resultNode;
  57. }
  58. System.out.println("后序比较~");
  59. if(this.id == findId) {
  60. return this;//进行判断值进行查找
  61. }
  62. return resultNode;
  63. }

二叉树类

说明:这里只是简单的判断根节点是否为空之后就对根节点进行对应的查找方法调用

  1. //前序遍历查找
  2. public TreeNode preOrderSearch(int findId) {
  3. if(root != null) {
  4. return root.preOrderSearch(findId);
  5. }else {
  6. System.out.println("二叉树为空无法查找!");
  7. return null;
  8. }
  9. }
  10. // 中序遍历查找
  11. public TreeNode infixOrderSearch(int findId) {
  12. if (root != null) {
  13. return root.infixOrderSearch(findId);
  14. } else {
  15. System.out.println("二叉树为空无法查找!");
  16. return null;
  17. }
  18. }
  19. // 后序遍历查找
  20. public TreeNode postOrderSearch(int findId) {
  21. if (root != null) {
  22. return root.postOrderSearch(findId);
  23. } else {
  24. System.out.println("二叉树为空无法查找!");
  25. return null;
  26. }
  27. }

测试类

  1. package tree;
  2. public class BinaryTree {
  3. public static void main(String[] args) {
  4. //创建一个二叉树
  5. MyBinaryTree tree = new MyBinaryTree();
  6. //创建四个节点
  7. TreeNode root = new TreeNode("宋江", 1);
  8. TreeNode node2 = new TreeNode("吴用", 2);
  9. TreeNode node3 = new TreeNode("卢俊", 3);
  10. TreeNode node4 = new TreeNode("林冲", 4);
  11. TreeNode node5 = new TreeNode("关胜", 5);
  12. //将node节点存放到root中
  13. root.setLeft(node2);
  14. root.setRight(node3);
  15. node3.setRight(node4);
  16. node3.setLeft(node5);
  17. //将root存放到二叉树中
  18. tree.setRoot(root);
  19. //进行前序遍历
  20. System.out.println("前序遍历:");
  21. tree.preOrder();
  22. System.out.println("中序遍历:");
  23. tree.infixOrder();
  24. System.out.println("后序遍历:");
  25. tree.postOrder();
  26. System.out.println("------------------------");
  27. //开始前中后序遍历查找
  28. System.out.println("前序遍历查找:");
  29. TreeNode node = tree.preOrderSearch(5);
  30. if(node != null) {
  31. System.out.println("找到节点为:"+node);
  32. }else {
  33. System.out.println("未找到节点!");
  34. }
  35. System.out.println("------------------------");
  36. //开始中序遍历查找
  37. System.out.println("中序遍历查找:");
  38. TreeNode node1 = tree.infixOrderSearch(5);
  39. if (node1 != null) {
  40. System.out.println("找到节点为:" + node1);
  41. } else {
  42. System.out.println("未找到节点!");
  43. }
  44. System.out.println("------------------------");
  45. //开始中序遍历查找
  46. System.out.println("后序遍历查找:");
  47. TreeNode node6 = tree.postOrderSearch(5);
  48. if (node6 != null) {
  49. System.out.println("找到节点为:" + node6);
  50. } else {
  51. System.out.println("未找到节点!");
  52. }
  53. }
  54. }

运行结果
QQ截图20200827111725.jpg

6、删除节点

源代码:BinaryTree.java

思路分析

要求说明
1)如果删除的节点是叶子节点,则删除该节点
2) 如果删除的节点是非叶子节点,则删除该子树.
3) 测试,删除掉 5 号叶子节点 和 3 号子树.
QQ截图20200828100653.jpg

代码实现(完整过程+测试)

说明:节点类以及二叉树类都只有各自一个删除方法,二叉树用于处理root根节点的情况,节点类处理左右两边情况

节点类

  1. //删除节点操作
  2. public void delNodeById(int delId) {
  3. //首先是判断左子节点是否为空,并且左子节点的id是否为要查找的id,是的话就删除
  4. if(this.left != null && this.left.id == delId) {
  5. this.left = null;
  6. System.out.println("成功删除节点,其id值为:"+delId);
  7. return;//删除成功就结束
  8. }
  9. //首先是判断右子节点是否为空,并且右子节点的id是否为要查找的id,是的话就删除
  10. if(this.right != null && this.right.id == delId) {
  11. this.right = null;
  12. System.out.println("成功删除节点,其id值为:"+delId);
  13. return;//删除成功就结束
  14. }
  15. //运行到这里就说明该节点的左右孩子都不符合删除要求,那么进行递归操作
  16. if(this.left != null) {
  17. this.left.delNodeById(delId);
  18. }
  19. if(this.right != null) {
  20. this.right.delNodeById(delId);
  21. }
  22. }

二叉树类

  1. //删除指定id的节点
  2. public void delNodeById(int delId) {
  3. if(root != null) {
  4. //来判断根节点是否为要删除的节点
  5. if(root.getId() == delId) {
  6. root = null;
  7. }else {
  8. //不为要删除的节点,那么执行删除节点的方法
  9. root.delNodeById(delId);
  10. }
  11. }else {
  12. System.out.println("树为空,无法进行删除操作!");
  13. }
  14. }

测试类

  1. package tree;
  2. public class BinaryTree {
  3. public static void main(String[] args) {
  4. //创建一个二叉树
  5. MyBinaryTree tree = new MyBinaryTree();
  6. //创建四个节点
  7. TreeNode root = new TreeNode("宋江", 1);
  8. TreeNode node2 = new TreeNode("吴用", 2);
  9. TreeNode node3 = new TreeNode("卢俊", 3);
  10. TreeNode node4 = new TreeNode("林冲", 4);
  11. TreeNode node5 = new TreeNode("关胜", 5);
  12. //将node节点存放到root中
  13. root.setLeft(node2);
  14. root.setRight(node3);
  15. node3.setRight(node4);
  16. node3.setLeft(node5);
  17. // //将root存放到二叉树中
  18. tree.setRoot(root);
  19. //
  20. //进行前序遍历
  21. System.out.println("前序遍历:");
  22. tree.preOrder();
  23. System.out.println();
  24. System.out.println("开始执行删除节点操作:");
  25. tree.delNodeById(5);//执行删除操作
  26. tree.delNodeById(3);//执行删除操作
  27. System.out.println();
  28. System.out.println("删除节点后前序遍历:");
  29. tree.preOrder();
  30. }

运行结果:
QQ截图20200828101532.jpg


二、顺序存储二叉树

1、顺序存储二叉树的概念

基本说明

从数据存储来看,数组存储方式和树的存储方式可以相互转换,即数组可以转换成树,树也可以转换成数组,
看右面的示意图:
QQ截图20200828102459.jpg

要求:

1) 右图的二叉树的结点,要求以数组的方式来存放 arr:[1,2,3,4,5,6,6]
2) 要求在遍历数组 arr 时,仍然可以以前序遍历,中序遍历和后序遍历的方式完成结点的遍

特点说明

说明:对一个左子节点以及右子节点、父节点的方式计算可以根据上面示例图来进行查看
QQ截图20200828102539.jpg

2、代码实现

说明:这里使用的是前序遍历的方法

  1. package tree;
  2. public class ArrayBinaryTreeDemo {
  3. public static void main(String[] args) {
  4. int[] arr = {1,2,3,4,5,6,7};
  5. ArrayBinaryTree tree = new ArrayBinaryTree(arr);
  6. //开始进行前序遍历
  7. tree.preOrder();
  8. }
  9. }
  10. class ArrayBinaryTree{
  11. private int[] arr;//用于存放数据节点的数据
  12. public ArrayBinaryTree(int[] arr) {
  13. super();
  14. this.arr = arr;
  15. }
  16. //进行重载调用有参数的方法
  17. public void preOrder() {
  18. preOrder(0);
  19. }
  20. //进行前序遍历
  21. public void preOrder(int index){
  22. //不符合条件的输出
  23. if(arr == null || arr.length == 0) {
  24. System.out.println("数组为空,二叉树无法遍历");
  25. return;
  26. }
  27. System.out.println(arr[index]);
  28. //向左递归 对应节点是否存在
  29. if((index*2+1)<arr.length) {
  30. preOrder(index*2+1);
  31. }
  32. //向右递归
  33. if((index*2+2)<arr.length) {
  34. preOrder(index*2+2);
  35. }
  36. }
  37. }

前中后序遍历如下:
QQ截图20200902113431.jpg

运行结果:
QQ截图20200902113543.jpg


整理者:长路 时间:2020/8/26-2020/8/28