树的基础知识

递归遍历

中序遍历

  1. public class InOrderTraversalRecursion {
  2. // 中序遍历
  3. public static List<Integer> inOrderTraversalRecursion(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. inOrderDFS(root, result);
  6. return result;
  7. }
  8. private static void inOrderDFS(TreeNode node, List<Integer> result) {
  9. if (node == null) {
  10. return;
  11. }
  12. inOrderDFS(node.left, result);
  13. result.add(node.val);
  14. inOrderDFS(node.right, result);
  15. }
  16. public static void main(String[] args) {
  17. TreeNode root = TreeNode.buildCBT();
  18. System.out.println(inOrderTraversalRecursion(root));
  19. }
  20. }

前序遍历

  1. public class PreOrderTraversalRecursion {
  2. // 前序遍历
  3. public static List<Integer> preOrderTraversalRecursion(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. preOrderDFS(root, result);
  6. return result;
  7. }
  8. private static void preOrderDFS(TreeNode node, List<Integer> result) {
  9. if (node == null) {
  10. return;
  11. }
  12. result.add(node.val);
  13. preOrderDFS(node.left, result);
  14. preOrderDFS(node.right, result);
  15. }
  16. public static void main(String[] args) {
  17. TreeNode root = TreeNode.buildCBT();
  18. System.out.println(preOrderTraversalRecursion(root));
  19. }
  20. }

后序遍历

  1. public class PostOrderTraversalRecursion {
  2. // 后遍历
  3. public static List<Integer> preOrderTraversalRecursion(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. preOrderDFS(root, result);
  6. return result;
  7. }
  8. private static void preOrderDFS(TreeNode node, List<Integer> result) {
  9. if (node == null) {
  10. return;
  11. }
  12. preOrderDFS(node.left, result);
  13. preOrderDFS(node.right, result);
  14. result.add(node.val);
  15. }
  16. public static void main(String[] args) {
  17. TreeNode root = TreeNode.buildCBT();
  18. System.out.println(preOrderTraversalRecursion(root));
  19. }
  20. }

迭代+栈

中序遍历

  1. public class InOrderTraversal {
  2. // 中序遍历
  3. public static List<Integer> inOrderTraversal(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. Stack<TreeNode> stack = new Stack<>();
  6. TreeNode cur = root;
  7. while (cur != null || !stack.isEmpty()) {
  8. // DFS
  9. while (cur != null) {
  10. stack.push(cur);
  11. cur = cur.left;
  12. }
  13. cur = stack.pop();
  14. // 中序遍历
  15. result.add(cur.val);
  16. // 处理右侧节点
  17. cur = cur.right;
  18. }
  19. return result;
  20. }
  21. public static void main(String[] args) {
  22. TreeNode root = TreeNode.buildCBT();
  23. System.out.println(inOrderTraversal(root));
  24. }
  25. }

前序遍历

  1. public class PreOrderTraversal {
  2. // 前序遍历
  3. public static List<Integer> preOrderTraversal(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. Stack<TreeNode> stack = new Stack<>();
  6. TreeNode cur = root;
  7. while (cur != null || !stack.isEmpty()) {
  8. // DFS
  9. while (cur != null) {
  10. // 前序遍历
  11. result.add(cur.val);
  12. // dfs
  13. stack.push(cur);
  14. cur = cur.left;
  15. }
  16. cur = stack.pop();
  17. // 处理右侧节点
  18. cur = cur.right;
  19. }
  20. return result;
  21. }
  22. public static void main(String[] args) {
  23. TreeNode root = TreeNode.buildCBT();
  24. System.out.println(preOrderTraversal(root));
  25. }
  26. }

后序遍历

  1. public class PostOrderTraversal {
  2. // 后续遍历
  3. public static List<Integer> postOrderTraversal(TreeNode root) {
  4. List<Integer> result = new ArrayList<>();
  5. Stack<TreeNode> stack = new Stack<>();
  6. TreeNode cur = root;
  7. TreeNode pre = null;
  8. while (cur != null || !stack.isEmpty()) {
  9. // DFS
  10. while (cur != null) {
  11. stack.push(cur);
  12. cur = cur.left;
  13. }
  14. // peek 栈顶,不是pop哦,
  15. cur = stack.peek();
  16. if (cur.right != null && cur.right != pre) {
  17. // 如果右子树还没被遍历,则外循环先遍历右子树
  18. cur = cur.right;
  19. } else {
  20. // 遍历当前节点
  21. result.add(cur.val);
  22. // 后序遍历,此时当前节点的子节点都遍历完了,可以pop掉
  23. stack.pop();
  24. // 更新pre节点
  25. pre = cur;
  26. // 置空cur节点,待下一轮循环pop出来
  27. cur = null;
  28. }
  29. }
  30. return result;
  31. }
  32. public static void main(String[] args) {
  33. TreeNode root = TreeNode.buildCBT();
  34. System.out.println(postOrderTraversal(root));
  35. }
  36. }

二叉树的深度优先搜索

47:二叉树剪枝

:::info 题目:一棵二叉树的所有节点的值要么是0要么是1,请剪除该二叉树中所有节点的值全都是0的子树。例如,在剪除图8.2(a)中二叉树中所有节点值都为0的子树之后的结果如图8.2(b)所示。
image.png :::

  1. public class PruneTree {
  2. public static TreeNode pruneTree(TreeNode root) {
  3. if (root == null) {
  4. return root;
  5. }
  6. // 递归处理左子树
  7. root.left = pruneTree(root.left);
  8. // 递归处理左子树
  9. root.right = pruneTree(root.right);
  10. // 如果当前节点是叶子节点,且值为0,那它可以被剪去
  11. if (root.left == null && root.right == null && root.val == 0) {
  12. // 从递归的视角看,
  13. // 由于是后续遍历,如果一个节点的子节点都是0,那么当递归遍历到自己的时候,它的子节点都已经被剪完了,也可以被剪去
  14. return null;
  15. }
  16. return root;
  17. }
  18. }

48:序列化和反序列化二叉树

:::info 题目:请设计一个算法将二叉树序列化成一个字符串,并能将该字符串反序列化出原来二叉树的算法。 :::

  1. public class SerializeBinaryTree {
  2. /**
  3. * 前序遍历二叉树,null节点用#补齐
  4. * @param root
  5. * @return
  6. */
  7. public static String serializeBinaryTree(TreeNode root) {
  8. if (root == null) {
  9. return "#";
  10. }
  11. String leftString = serializeBinaryTree(root.left);
  12. String rightString = serializeBinaryTree(root.right);
  13. return root.val + "," + leftString + "," + rightString;
  14. }
  15. public static TreeNode deserializeBinaryTree(String string) {
  16. String[] arr = string.split(",");
  17. int[] index = {0};
  18. return deserializeBinaryTree(arr, index);
  19. }
  20. /**
  21. * 由于是前序遍历, 所以当前节点就是父节点,然后再分别递归处理左子树和右子树
  22. * 递归的终止条件是遇到"#"空节点时,返回null
  23. * @param arr
  24. * @param index
  25. * @return
  26. */
  27. private static TreeNode deserializeBinaryTree(String[] arr, int[] index) {
  28. String val = arr[index[0]];
  29. index[0]++;
  30. if (val.equals("#")) {
  31. return null;
  32. }
  33. TreeNode node = new TreeNode(Integer.valueOf(val));
  34. node.left = deserializeBinaryTree(arr, index);
  35. node.right = deserializeBinaryTree(arr, index);
  36. return node;
  37. }
  38. }

49:从根节点到叶节点的路径数字之和

:::info 题目:在一棵二叉树中所有节点都在0~9的范围之内,从根节点到叶节点的路径表示一个数字。求二叉树中所有路径表示的数字之和。例如,图8.4的二叉树有3条从根节点到叶节点的路径,它们分别表示数字395、391和302,这3个数字之和是1088。 :::

  1. public class PathSum {
  2. public static int pathSum(TreeNode root) {
  3. return pathSum(0, root);
  4. }
  5. public static int pathSum(int sum, TreeNode node) {
  6. if (node == null) {
  7. return 0;
  8. }
  9. sum = sum * 10 + node.val;
  10. // 如果是叶子节点,此时要终止。
  11. // 上一句已经加过了,不然走到下面又多加一次
  12. // PS:递归的理解可真难orz
  13. if (node.left == null && node.right == null) {
  14. return sum;
  15. }
  16. // sum是int,是值传递,所以不会影响其他节点的递归
  17. return pathSum(sum, node.left) + pathSum(sum, node.right);
  18. }
  19. }

50:向下的路径节点值之和

:::info 题目:给定一棵二叉树和一个值sum,求二叉树中节点值之和等于sum的路径的数目。路径的定义为二叉树中顺着指向子节点的指针向下移动所经过的节点,但不一定从根节点开始,也不一定到叶节点结束。例如,在如图8.5所示中的二叉树中有两条路径的节点值之和等于8,其中,第1条路径从节点5开始经过节点2到达节点1,第2条路径从节点2开始到节点6。
image.png :::

  1. public class pathSum50 {
  2. public static int pathsum(TreeNode root, int sum) {
  3. // 累加和,count
  4. Map<Integer, Integer> map = new HashMap<>();
  5. map.put(0, 1);
  6. int pathSum = 0;
  7. return dfs(root, map, sum, pathSum);
  8. }
  9. private static int dfs(TreeNode node, Map<Integer, Integer> map, int sum, int pathSum) {
  10. if (node == null) {
  11. return 0;
  12. }
  13. pathSum += node.val;
  14. int count = map.getOrDefault(pathSum - sum, 0);
  15. map.put(pathSum, map.getOrDefault(pathSum, 0) + 1);
  16. count += dfs(node.left, map, sum, pathSum);
  17. count += dfs(node.right, map, sum, pathSum);
  18. // 此时当前节点的左右字数都遍历完了,count已经计算上了
  19. // 就把自己的累计值从map里面去掉,这样不影响其他路径的计算
  20. map.put(pathSum, map.get(pathSum) - 1);
  21. return count;
  22. }
  23. }

51:节点值之和最大的路径

:::info 题目:在二叉树中将路径定义为顺着节点之间的连接从任意一个节点开始到达任意一个节点所经过的所有节点。路径中至少包含一个节点,不一定经过二叉树的根节点,也不一定经过叶节点。给定非空的一棵二叉树,请求出二叉树所有路径上节点值之和的最大值。例如,在如图8.6所示的二叉树中,从节点15开始经过节点20到达节点7的路径的节点值之和为42,是节点值之和最大的路径。 :::

// TODO

二叉搜索树

// TODO

TreeSet和TreeMap的应用