• [模版总结] - 树的基本算法1 - 遍历


    树结构定义

    一种非线性存储结构,具有存储“一对多”关系的数据元素集合

    种类

    • General Tree
      • Trie
      • B/B+ 树
    • 二叉树
      • 满/完满/完全二叉树
        • 完美BT : 除了叶子结点外所有节点都有两个字节点,每一层都完满填充
        • 完全BT: 除最后一层以外其他每一层都完美填充,最后一层从左到右紧密填充
        • 完满BT:  除了叶子结点外所有节点都有两个字节点
      • 二叉搜索树 BST
        • 平衡BST 
          • 红黑树
          • 伸展树
          • 自平衡二叉查找树AVL
          • 替罪羊树
      • 线索二叉树
      • 霍夫曼树/最优二叉树

    二叉树遍历方式

    所有二叉树基本遍历时间复杂度均为:O(N), N代表结点数量。

    前序遍历 (根左右)

    递归写法 

    由于前序的特性,他可以展示树结构的继承关系,所以通常前序遍历会用在复制/打印树结构,比如序列化/反序列化,打印文件系统结构。

    1. class Solution {
    2. public List preorderTraversal(TreeNode root) {
    3. List res = new ArrayList<>();
    4. dc(root, res);
    5. return res;
    6. }
    7. private void dc(TreeNode root, List res) {
    8. if (root==null) return;
    9. res.add(root.val);
    10. dc(root.left, res);
    11. dc(root.right, res);
    12. }
    13. }

    中序遍历 (左根右)

    递归写法

    中序遍历最常用就是打印BST结构

    1. class Solution {
    2. List res;
    3. public List inorderTraversal(TreeNode root) {
    4. res = new ArrayList<>();
    5. dc(root);
    6. return res;
    7. }
    8. private void dc(TreeNode root) {
    9. if (root==null) return;
    10. dc(root.left);
    11. res.add(root.val);
    12. dc(root.right);
    13. }
    14. }

    后续遍历 (左右根) 

    后续遍历由于特性是先搜索叶子结点,所以可以用来做一些叶子结点操作(删除叶子结点),还有一些归并操作(计算算术表达式)

    1. class Solution {
    2. public List postorderTraversal(TreeNode root) {
    3. List res = new ArrayList<>();
    4. dc(root, res);
    5. return res;
    6. }
    7. private void dc(TreeNode root, List res) {
    8. if (root==null) return;
    9. dc(root.left, res);
    10. dc(root.right, res);
    11. res.add(root.val);
    12. }
    13. }

    层级遍历 I - 自上而下

    树/图类层级遍历直接BFS即可

    1. class Solution {
    2. public List> levelOrder(TreeNode root) {
    3. List> res = new ArrayList<>();
    4. if (root==null) return res;
    5. Queue q = new LinkedList<>();
    6. q.add(root);
    7. while (!q.isEmpty()) {
    8. int size = q.size();
    9. List tmp = new ArrayList<>();
    10. for (int i=0; i
    11. TreeNode curr = q.poll();
    12. tmp.add(curr.val);
    13. if (curr.left!=null) q.add(curr.left);
    14. if (curr.right!=null) q.add(curr.right);
    15. }
    16. res.add(tmp);
    17. }
    18. return res;
    19. }
    20. }

    层级遍历 II - 自下而上

    1. class Solution {
    2. public List> levelOrderBottom(TreeNode root) {
    3. List> res = new ArrayList<>();
    4. if (root==null) return res;
    5. Queue q = new LinkedList<>();
    6. q.add(root);
    7. while (!q.isEmpty()) {
    8. int size = q.size();
    9. List tmp = new ArrayList<>();
    10. for (int i=0; i
    11. TreeNode curr = q.poll();
    12. if (curr.left!=null) q.add(curr.left);
    13. if (curr.right!=null) q.add(curr.right);
    14. tmp.add(curr.val);
    15. }
    16. res.add(0, tmp);
    17. }
    18. return res;
    19. }
    20. }

    ZigZag 遍历

    1. class Solution {
    2. public List> zigzagLevelOrder(TreeNode root) {
    3. List> res = new ArrayList<>();
    4. dfs(root, 0, res);
    5. return res;
    6. }
    7. private void dfs(TreeNode root, int height, List> res) {
    8. if (root==null) return;
    9. if (res.size()<=height) res.add(new ArrayList<>());
    10. if (height%2==0) {
    11. res.get(height).add(root.val);
    12. } else {
    13. res.get(height).add(0, root.val);
    14. }
    15. dfs(root.left, height+1, res);
    16. dfs(root.right, height+1, res);
    17. }
    18. }

    一些特别的遍历: 

    逐列遍历 

    T: O(N + C\times RlogR) , N表示dfs遍历时间复杂度,C表示列数,R表示每一列的行数

    1. class Solution {
    2. TreeMap>> colMap;
    3. public List> verticalOrder(TreeNode root) {
    4. if (root==null) return new ArrayList<>();
    5. colMap = new TreeMap<>();
    6. dfs(root, 0, 0);
    7. List> res = new ArrayList<>();
    8. for (int idx: colMap.keySet()) {
    9. Collections.sort(colMap.get(idx), (a, b) -> {
    10. return a.getKey()-b.getKey();
    11. });
    12. List tmp = new ArrayList<>();
    13. for (Pair a: colMap.get(idx)) {
    14. tmp.add(a.getValue());
    15. }
    16. res.add(tmp);
    17. }
    18. return res;
    19. }
    20. private void dfs(TreeNode root, int row, int col) {
    21. if (root==null) return;
    22. colMap.putIfAbsent(col, new ArrayList<>());
    23. colMap.get(col).add(new Pair<>(row, root.val));
    24. dfs(root.left, row+1, col-1);
    25. dfs(root.right, row+1, col+1);
    26. }
    27. }

  • 相关阅读:
    Java项目:SSM在线个人PC电脑商城平台网站系统
    VUE 笔记 基础语法篇
    手持振弦采集仪VH03各种接口使用说明
    OpenHarmony中Element类是什么?
    理解「跨域(源)」和「跨网站」的差异与联系
    软件测试工程师怎么样面试上好的公司?
    Go语言开发环境搭建指南:快速上手构建高效的Go开发环境
    手把手教你如何采用服务商模式实现微信支付
    Leetcode 第320场周赛
    Matlab图像处理-
  • 原文地址:https://blog.csdn.net/ok1382038/article/details/134323975