2025高频面试算法总结篇【二叉树】

news2025/4/18 15:45:05

文章目录

  • 直接刷题链接直达
  • 非递归实现求二叉树的深度
  • 非递归从左至右打印一颗二叉树中的所有路径
  • 判断平衡二叉树
  • 二叉搜索树中第K小的元素
  • 二叉树的完全性检验
  • 根据前&中序遍历结果重建二叉树
  • 二叉树的最近公共祖先
  • 二叉树的直径
  • 二叉树的遍历


直接刷题链接直达

  • 非递归实现求二叉树的深度
    • 引入队列,统计当前层次节点数目,逐层遍历
    • 二叉树的深度(递归和非递归)
    • 102. 二叉树的层次遍历
  • 非递归从左至右打印一颗二叉树中的所有路径
    • 257. 二叉树的所有路径
  • 判断平衡二叉树
    • 110. 平衡二叉树
  • 寻找二叉搜索树中第一个大于k的节点
    • 中序遍历求值
    • 类似于 230. 二叉搜索树中第K小的元素
  • 二叉树的完全性检验
    • 958. 二叉树的完全性检验
  • 根据前&中序遍历结果重建二叉树
    • 首先找到根节点,再划分左右子树区域,逐层递归找到左右子节点
    • 105. 从前序与中序遍历序列构造二叉树
  • 实现一个二叉搜索树迭代器,包括 next() hasNext() 方法(PayPal)
    • 173. 二叉搜索树迭代器
  • 二叉树的右视图(Shopee)
    • 199. 二叉树的右视图
  • 给定一个二叉树根节点和指定路径,判断二叉树中是否存在给定指定路径,且要求指定路径最后一个元素为叶节点
  • 将一颗二叉搜索树转化成一个排序的双向链表
    • 不能创建新结点
    • 二叉搜索树与双向链表
  • 二叉树的最近公共祖先
    • 首先判断当前节点是否为指定结点,是则返回
    • 递归当前结点的左右子树
    • 如果两边均不为null,表示找到,返回当前结点,均为null则返回null,否则返回对应子节点(left / right)
    • 236. 二叉树的最近公共祖先
    • Lowest Common Ancestor Binary Tree(各种情况详细举例,推荐)
  • 二叉树的中序遍历的下一个结点
    • 先判断右子结点不为空,返回右子节点最左边的节点
    • 若父节点不为空,上溯到根节点的 “/”即返回
    • 二叉树的下一个结点
  • 红黑树描述及其复杂度分析(插入/查找)
    • 查找、插入、删除时间复杂度 --> O(logN)
    • 红黑树 Wiki
    • 26 | 红黑树(下):掌握这些技巧,你也可以实现一个红黑树 (红黑树分析)
  • 二叉树的直径
    • 543. 二叉树的直径
  • 二叉树的遍历
    • 二叉树的前序遍历 / 二叉树的中序遍历 / 二叉树的后序遍历

非递归实现求二叉树的深度

题目: 给定一个二叉树 root ,返回其最大深度。
二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。

递归:

class Solution {
    public int maxDepth(TreeNode root) {
        if (root == null) return 0;
        int left = maxDepth(root.left);
        int right = maxDepth(root.right);
        return (left > right ? left:right) +1;
    }
}

非递归:
引入队列,统计当前层次节点数目,逐层遍历

class Solution {
    public int maxDepth(TreeNode root) {
        if (root == null) return 0;
        Queue<TreeNode> q = new LinkedList<>();
        q.offer(root);
        int depth = 0;
        while (!q.isEmpty()) {
            depth++;
            int total = q.size(); // 统计这层的数量=》全部出队列,然后他们的子节点入队
            while (total-- > 0) {
                TreeNode temp = q.poll();
                if (temp.left != null) {
                    q.offer(temp.left);
                }
                if (temp.right != null) {
                    q.offer(temp.right);
                }
            }
        }

        return depth;

    }
}

非递归从左至右打印一颗二叉树中的所有路径

题目: 给你一个二叉树的根节点 root ,按 任意顺序 ,返回所有从根节点到叶子节点的路径。

递归:

class Solution {

    public List<String> binaryTreePaths(TreeNode root) {
        treePath(root, "");
        return res;
    }
    List<String> res = new ArrayList<>();
    public void treePath(TreeNode root, String path) {
        if (root == null) {
            return;
        }
        // 叶子节点
        if (root.left == null && root.right == null) {
            res.add(path+root.val);
            return;
        }

        treePath(root.left, path+root.val+"->");

        treePath(root.right, path+root.val+"->");
    }
}

非递归:

class Solution {
    // 257. 二叉树的所有路径
    public List<String> binaryTreePaths(TreeNode root) {
        List<String> res = new LinkedList<>();
        if (root == null) return res;
        // 存放 节点
        Stack<TreeNode> nodeStack = new Stack<>();
        // 存放 path
        Stack<String> pathStack = new Stack<>();
        // 根节点 入栈
        nodeStack.push(root);
        pathStack.push(root.val + "");
        while (!nodeStack.isEmpty()) {
            TreeNode node = nodeStack.pop();
            String path = pathStack.pop();
            if (node.left == null && node.right == null) {
                res.add(path);
            }
            if (node.right != null) {
                pathStack.push(path + "->" + node.right.val);
                nodeStack.push(node.right);
            }
            
            if (node.left != null) {
                pathStack.push(path + "->" + node.left.val);
                nodeStack.push(node.left);
            }
            
        }
        return res;
    }
}


判断平衡二叉树

题目:给定一个二叉树,判断它是否是 平衡二叉树

class Solution {
    
    public boolean isBalanced(TreeNode root) {
        if (root == null) return true;
        if(Math.abs(findDepth(root.left)-findDepth(root.right)) > 1) return false;
        return isBalanced(root.left) && isBalanced(root.right);
    }

    public int findDepth(TreeNode root) {
        if (root == null) return 0;

        int left = findDepth(root.left);
        int right = findDepth(root.right);

        return (left > right ? left:right)+1;
    }
}

二叉搜索树中第K小的元素

题目: 给定一个二叉搜索树的根节点 root ,和一个整数 k ,请你设计一个算法查找其中第 k 小的元素(从 1 开始计数)。

class Solution {
    List<Integer> list;
    public int kthSmallest(TreeNode root, int k) {
        list = new ArrayList<>();
        inOrder(root);
        return list.get(k-1);
    }
    public void inOrder(TreeNode root) {
        if (root == null) return;
        inOrder(root.left);
        list.add(root.val);
        inOrder(root.right);
    }

}

二叉树的完全性检验

题目: 给你一棵二叉树的根节点 root ,请你判断这棵树是否是一棵 完全二叉树 。

对于一个完全二叉树,层序遍历的过程中遇到第一个空节点之后不应该再出现非空节点


class Solution {
    public boolean isCompleteTree(TreeNode root) {
        if (root == null) return true;
        // 层次遍历
        Queue<TreeNode> q = new LinkedList<>();
        q.offer(root);
        boolean flag = false; // 开始出现 null 节点
        while(!q.isEmpty()) {
            TreeNode t = q.poll();
            if (flag && t != null) {
                return false;
            }

            if (t == null) {
                flag = true;
            }else {
                q.offer(t.left);
                q.offer(t.right);
            }

            


        }

        return true;
        
    }
}

根据前&中序遍历结果重建二叉树

题目:给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。

class Solution {
    private HashMap<Integer, Integer> inorderMap; // 存放中序值和索引
    private int preorderIndex = 0;// 当前访问的前序的位置
    public TreeNode buildTree(int[] preorder, int[] inorder) {
        inorderMap = new HashMap<>();
        for (int i = 0; i < inorder.length; i++) {
            inorderMap.put(inorder[i], i);
        }

        return createTree(preorder, 0, inorder.length-1);

    }

    public TreeNode createTree(int[] preorder, int left, int right) {
        if (left > right) return null;
        int val = preorder[preorderIndex++];

        TreeNode root = new TreeNode(val);
        root.left = createTree(preorder, left, inorderMap.get(val)-1);
        root.right = createTree(preorder, inorderMap.get(val)+1, right);
        
        return root;
    }
}

二叉树的最近公共祖先

题目: 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。

百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”

class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        if (root == null || p == root || q == root) return root;
        TreeNode left = lowestCommonAncestor(root.left, p, q);
        TreeNode right = lowestCommonAncestor(root.right, p, q);

        if (left != null && right != null) return root;
        return left != null ? left:right;
    }
}

二叉树的直径

给你一棵二叉树的根节点,返回该树的 直径 。

二叉树的 直径 是指树中任意两个节点之间最长路径的 长度 。这条路径可能经过也可能不经过根节点 root 。

两节点之间路径的 长度 由它们之间边数表示。

class Solution {
    // 543 二叉树的直径
    private int maxDiameter = 0;

    public int diameterOfBinaryTree(TreeNode root) {
        maxDepth(root); // 在计算深度的过程中更新最大直径
        return maxDiameter;
    }

    // 计算树的最大深度,同时更新最大直径
    private int maxDepth(TreeNode node) {
        if (node == null) {
            return 0;
        }
        int left = maxDepth(node.left); // 左子树深度
        int right = maxDepth(node.right); // 右子树深度

        // 更新直径:左子树深度 + 右子树深度
        maxDiameter = Math.max(maxDiameter, left + right);

        // 返回当前节点的深度
        return Math.max(left, right) + 1;
    }


    
    
}

二叉树的遍历

前序遍历:

class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        preorder(root);
        return ans;
    }

    List<Integer> ans = new ArrayList<>();
    public void preorder(TreeNode root) {
        if (root == null) return;
        ans.add(root.val);
        preorder(root.left);
        preorder(root.right);
    }
}

中序遍历:

class Solution {
    // 92 中序遍历
    public List<Integer> inorderTraversal(TreeNode root) {
        inorder(root);
        return ans;
    }
    List<Integer> ans = new ArrayList<>();
    public void inorder(TreeNode root) {
        if (root == null) return;
        inorder(root.left);
        ans.add(root.val);
        inorder(root.right);
    }
}

后续遍历:

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        postorder(root);
        return ans;
    }
    List<Integer> ans = new ArrayList<>();
    public void postorder(TreeNode root) {
        if (root == null) return;
        postorder(root.left);
        postorder(root.right);
        ans.add(root.val);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2330021.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何理解神经网络中的“分段线性单元”,优雅解析前向和反向传播

什么是非线性 非线性本质上指的是一个系统或函数中输入与输出之间的关系不呈现简单的比例关系&#xff0c;也就是说&#xff0c;输出不只是输入的线性组合 ( 比如 y k 1 x 1 k 2 x 2 b ) (比如yk1x1k2x2b) (比如yk1x1k2x2b)。下面详细解释这个概念&#xff1a; 缺乏叠加性…

WVP-GB28181摄像头管理平台存在弱口令

免责声明&#xff1a;本号提供的网络安全信息仅供参考&#xff0c;不构成专业建议。作者不对任何由于使用本文信息而导致的直接或间接损害承担责任。如涉及侵权&#xff0c;请及时与我联系&#xff0c;我将尽快处理并删除相关内容。 漏洞描述 攻击者可利用漏洞获取当前系统管…

开源身份和访问管理方案之keycloak(三)keycloak健康检查(k8s)

文章目录 开源身份和访问管理方案之keycloak&#xff08;三&#xff09;keycloak健康检查启用运行状况检查 健康检查使用Kubernetes下健康检查Dockerfile 中 HEALTHCHECK 指令 健康检查Docker HEALTHCHECK 和 Kubernetes 探针 开源身份和访问管理方案之keycloak&#xff08;三&…

Android学习总结之service篇

引言 在 Android 开发里&#xff0c;Service 与 IntentService 是非常关键的组件&#xff0c;它们能够让应用在后台开展长时间运行的操作。不过&#xff0c;很多开发者仅仅停留在使用这两个组件的层面&#xff0c;对其内部的源码实现了解甚少。本文将深入剖析 Service 和 Inte…

spring mvc异步请求 sse 大文件下载 断点续传下载Range

学习连接 异步Servlet3.0 Spring Boot 处理异步请求&#xff08;DeferredResult 基础案例、DeferredResult 超时案例、DeferredResult 扩展案例、DeferredResult 方法汇总&#xff09; spring.io mvc Asynchronous Requests 官网文档 spring.io webflux&webclient官网文…

Opencv计算机视觉编程攻略-第十节 估算图像之间的投影关系

目录 1. 计算图像对的基础矩阵 2. 用RANSAC 算法匹配图像 3. 计算两幅图像之间的单应矩阵 4. 检测图像中的平面目标 图像通常是由数码相机拍摄的&#xff0c;它通过透镜投射光线成像&#xff0c;是三维场景在二维平面上的投影&#xff0c;这表明场景和它的图像之间以及同一…

14.流程自动化工具:n8n和家庭自动化工具:node-red

n8n 安装 docker方式 https://docs.n8n.io/hosting/installation/docker/ #https://hub.docker.com/r/n8nio/n8n docker pull n8nio/n8n:latest docker rm -f n8n; docker run -it \ --network macvlan --hostname n8n \ -e TZ"Asia/Shanghai" \ -e GENERIC_TIME…

图形渲染: tinyrenderer 实现笔记(Lesson 1 - 4)

目录 项目介绍环境搭建Lesson 1: Bresenham’s Line Drawing Algorithm&#xff08;画线算法&#xff09;Lesson 2: Triangle rasterization 三角形光栅化Scanline rendering 线性扫描Modern rasterization approach 现代栅格化方法back-face culling 背面剔除 Lesson 3: Hidde…

大规模硬件仿真系统的编译挑战

引言&#xff1a; 随着集成电路设计复杂度的不断提升&#xff0c;硬件仿真系统在现代芯片设计流程中扮演着越来越重要的角色。基于FPGA&#xff08;现场可编程门阵列&#xff09;的商用硬件仿真系统因其灵活性、全自动化、高性能和可重构性&#xff0c;成为验证大规模集成电路设…

记一次常规的网络安全渗透测试

目录&#xff1a; 前言 互联网突破 第一层内网 第二层内网 总结 前言 上个月根据领导安排&#xff0c;需要到本市一家电视台进行网络安全评估测试。通过对内外网进行渗透测试&#xff0c;网络和安全设备的使用和部署情况&#xff0c;以及网络安全规章流程出具安全评估报告。本…

【8】搭建k8s集群系列(二进制部署)之安装work-node节点组件(kubelet)

一、下载k8s二进制文件 下载地址&#xff1a; https://github.com/kubernetes/kubernetes/blob/master/CHANGELOG/CHANGELOG -1.20.md 注&#xff1a;打开链接你会发现里面有很多包&#xff0c;下载一个 server 包就够了&#xff0c;包含了 Master 和 Worker Node 二进制文件。…

使用 VIM 编辑器对文件进行编辑

一、VIM 的两种状态 VIM&#xff08;vimsual&#xff09;是 Linux/UNIX 系列 OS 中通用的全屏编辑器。vim 分为两种状态&#xff0c;即命令状态和编辑状态&#xff0c;在命令状态下&#xff0c;所键入的字符系统均作命令来处理&#xff1b;而编辑状态则是用来编辑文本资料&…

visual studio 2022的windows驱动开发

在visual studio2022中&#xff0c;若在单个组件中找不到Windows Driver Kit (WDK)选项&#xff0c;可通过提升vs版本解决&#xff0c;在首次选择时选择WDM 创建好项目在Source Files文件夹中创建一个test.c文件&#xff0c;并输入以下测试代码&#xff1a; #include <ntdd…

基于大数据的美团外卖数据可视化分析系统

【大数据】基于大数据的美团外卖数据可视化分析系统 &#xff08;完整系统源码开发笔记详细部署教程&#xff09;✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 该系统通过对海量外卖数据的深度挖掘与分析&#xff0c;能够为美团外卖平台提供运营决策支…

vue2打包部署到nginx,解决路由history模式下页面空白问题

项目使用的是vue2&#xff0c;脚手架vue-cli 4。 需求&#xff1a;之前项目路由使用的是hash&#xff0c;现在要求调整为history模式&#xff0c;但是整个过程非常坎坷&#xff0c;遇到了页面空白问题。现在就具体讲一下这个问题。 首先&#xff0c;直接讲路由模式由hash改为…

【数据结构】排序算法(中篇)·处理大数据的精妙

前引&#xff1a;在进入本篇文章之前&#xff0c;我们经常在使用某个应用时&#xff0c;会出现【商品名称、最受欢迎、购买量】等等这些榜单&#xff0c;这里面就运用了我们的排序算法&#xff0c;作为刚学习数据结构的初学者&#xff0c;小编为各位完善了以下几种排序算法&…

AI随身翻译设备:从翻译工具到智能生活伴侣

文章目录 AI随身翻译设备的核心功能1. 实时翻译2. 翻译策略3. 翻译流程4. 输出格式 二、AI随身翻译设备的扩展功能1. 语言学习助手2. 旅行助手3. 商务助手4. 教育助手5. 健康助手6. 社交助手7. 技术助手8. 生活助手9. 娱乐助手10. 应急助手 三、总结四、未来发展趋势&#xff0…

chromadb 安装和使用

简介 Chromadb 是一个开源的嵌入式向量数据库&#xff0c;专为现代人工智能和机器学习应用设计&#xff0c;旨在高效存储、检索和管理向量数据。以下是关于它的详细介绍&#xff1a; 核心特性 易于使用&#xff1a;提供了简洁直观的 API&#xff0c;即使是新手也能快速上手…

LabVIEW 在故障诊断中的算法

在故障诊断领域&#xff0c;LabVIEW 凭借其强大的图形化编程能力、丰富多样的工具包以及卓越的功能性能&#xff0c;成为工程师们进行故障诊断系统开发的得力助手。通过运用各种算法&#xff0c;能够对采集到的信号进行全面、深入的分析处理&#xff0c;从而准确地诊断出系统中…

springboot 启动方式 装配流程 自定义starter 文件加载顺序 常见设计模式

目录 springboot介绍 核心特性 快速搭建 Spring Boot 项目 方式一&#xff1a;使用 Spring Initializr 方式二&#xff1a;使用 IDE 插件 示例代码 1. 创建项目并添加依赖 2. 创建主应用类 3. 创建控制器类 4. 运行应用程序 配置文件 部署和监控 部署 监控 与其…