news 2026/4/16 6:54:17

从零开始写算法——二叉树篇3:对称二叉树 + 二叉树直径

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
从零开始写算法——二叉树篇3:对称二叉树 + 二叉树直径

二叉树是数据结构中递归思想的最佳演练场。很多同学在刷题时,往往看答案觉得很简单,自己写却无从下手。

其实,二叉树的递归无非就两种套路:

  1. 怎么传下去?(父节点把要求传给子节点,如判断对称)

  2. 怎么收上来?(子节点把结果汇报给父节点,如计算直径)

今天我们通过两道经典题目(LeetCode 101 和 543),来彻底搞懂这两个套路。


01. 对称二叉树 (Symmetric Tree)

题目解析

我们要判断一棵树是不是轴对称的。这就好比我们在树的中间放一面镜子,左边的样子和右边的样子必须完全重合。

核心逻辑:镜像对比

很多初学者容易犯的错误是:直接判断root->leftroot->right是不是“完全相同”的树。错!对称不是“相同”,而是“镜像”。

我们可以把这棵大树拆解成无数个小的“三角单元”。对于任意两个对称位置的节点p(左边那个)和q(右边那个),如果要满足对称,必须同时满足三个条件:

  1. 值相等p的值必须等于q的值。

  2. 外侧对齐p的左孩子(最外侧)必须等于q的右孩子(最外侧)。

  3. 内侧对齐p的右孩子(靠近中间)必须等于q的左孩子(靠近中间)。

代码深度拆解

C++代码实现:

class Solution { public: bool isSymmetric(TreeNode* root) { // 根节点的左子树和右子树进行镜像对比 return isSameTree(root->left, root->right); } // 这里虽然函数名叫 isSameTree,但实际逻辑是 checkMirror // 作用:判断 p 和 q 两棵树是否互为镜像 bool isSameTree(TreeNode* p, TreeNode* q) { // 1. 递归终止条件:处理空节点的情况 // 如果两个都为空,说明到底了且匹配,返回 true // 如果只有一个为空,说明不对称(结构不同),返回 false if (p == nullptr || q == nullptr) return q == p; // 2. 核心递归逻辑 // 条件A: 当前节点值必须一样 // 条件B: p的左边 VS q的右边 (外侧互相对比) // 条件C: p的右边 VS q的左边 (内侧互相对比) return p->val == q->val && isSameTree(p->left, q->right) && isSameTree(p->right, q->left); } };

为什么这么写?代码中最精髓的一句是isSameTree(p->left, q->right) && isSameTree(p->right, q->left)。这正是体现了**“镜像”**的定义。如果你写成了p->left, q->left,那就是判断两棵树是否完全一样(平移),而不是对称了。

复杂度分析

  • 时间复杂度O(N)。 我们需要遍历树中的每一个节点一次。

  • 空间复杂度O(H),其中 H 是树的高度。 这里的空间消耗主要来自递归调用的栈空间。在最坏情况下(树退化成链表),高度为 N;最好情况下(完全二叉树),高度为 logN。


02. 二叉树的直径 (Diameter of Binary Tree)

题目解析

这道题要求的是二叉树中任意两个节点间的最长路径长度。注意一个大坑:这条最长路径不一定经过根节点。它可能完全存在于左子树内部,也可能在右子树内部。

核心逻辑:后序遍历 + 全局打擂台

既然路径可能出现在任何地方,但所有的路径都有一个共同点:它一定有一个“最高点”(转折点)。 对于任何一个节点,穿过该节点的最长路径 =左子树的最大深度+右子树的数据深度

所以我们的策略是:

  1. 自底向上(后序遍历):先算出子树的深度,再汇报给父节点。

  2. 全局打擂:在计算深度的过程中,顺便计算“经过当前节点的最长路径”,并用一个全局变量ans记录历史最大值。

代码深度拆解

C++代码实现:

class Solution { // 全局变量维护最大直径 // 因为最长路径可能不经过根节点,所以需要在遍历过程中实时更新最大值 int ans = 0; // 这个函数的定义非常关键: // 它的作用是:返回以 root 为根的树的【最大深度】 // 它的副作用是:更新经过 root 的【最大直径】 int dfs(TreeNode* root) { // 1. 终止条件:空节点深度为 0 if (root == nullptr) return 0; // 2. 递归获取左右子树的深度(自底向上) int left = dfs(root->left); int right = dfs(root->right); // 3. 【核心逻辑之一:更新答案】 // 经过当前节点的最长路径 = 左臂长 + 右臂长 // 这是一条"穿过"当前节点的路径,类似倒"V"字型 ans = max(ans, left + right); // 4. 【核心逻辑之二:返回深度】 // 当前节点要汇报给父节点的是:我这条分支有多长? // 我只能选左边或右边更长的一条路继续往上延伸,不能两条都选(那样就分叉了) // 所以返回 max(left, right) + 1 (加1是加上当前节点自己) return max(left, right) + 1; } public: int diameterOfBinaryTree(TreeNode* root) { dfs(root); return ans; } };

为什么这么写?很多同学搞混淆的地方在于:为什么更新ans用加法,而返回值用max

  • 更新ans:我们在找直径(路径长度)。一条路径穿过当前节点,是把左腿和右腿连起来,所以是left + right

  • 返回值:我们在算深度(高度)。对于父节点来说,当前节点只是它的一条“腿”。一条路径不能分叉,只能选最长的那条腿往上走,所以是max(left, right) + 1

复杂度分析

  • 时间复杂度O(N)。 每个节点被访问一次,计算深度是 O(1) 的操作。

  • 空间复杂度O(H)。 同样取决于递归栈的深度。最坏情况 O(N),平均情况 O(logN)。


总结

这两道题虽然都是递归,但方向截然不同:

  1. 对称二叉树带着任务往下跑:拿着左边的节点去和右边匹配,这是一种自顶向下的逻辑。

  2. 二叉树直径带着结果往上报:子节点算好了深度,汇报给父节点,父节点顺便计算一下经过自己的最长路径,这是一种自底向上的逻辑。

掌握了这两种思维,你对二叉树的递归理解就上了一个台阶!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/9 0:15:26

图像生成大模型

推荐的图像生成大模型有豆包、即梦、腾讯云智能图像创作平台等。选择任意图像生成大模型平台体验其在生成人物摄影图像、风景植物图像、建筑设计图像、中国风格图像四种类型。序号类型任务提示词生成的图像1人物摄影生成婚礼上的新娘和伴娘示例:梦幻般的婚礼殿堂内&…

作者头像 李华
网站建设 2026/4/15 4:41:17

基于Dify镜像的RAG系统构建全流程演示

基于 Dify 镜像的 RAG 系统构建全流程解析 在企业加速拥抱 AI 的今天,一个现实问题摆在面前:如何让大语言模型真正“懂”自家业务?许多团队尝试过直接调用 GPT 或通义千问回答客户问题,结果往往不尽如人意——模型要么胡编乱造&am…

作者头像 李华
网站建设 2026/4/6 2:40:44

9、SharePoint关键设置与故障排除指南

SharePoint关键设置与故障排除指南 分布式缓存 在农场的每台服务器上运行相关操作后,可使用 Update-SPDistributedCacheSize cmdlet 更新大小。在SharePoint 2016中,安装时会应用带有CU7的App Fabric 1.1 for Windows Server服务,但垃圾收集不会自动配置,这点比较奇怪。…

作者头像 李华
网站建设 2026/4/13 15:46:31

21、SharePoint 工具与故障排除全解析

SharePoint 工具与故障排除全解析 1. SharePoint 管理器工具介绍 SharePoint 管理器工具是一款强大的故障排除利器。它当前不在 GitHub 上,可从 CodePlex(https://spm.codeplex.com )下载。下载应用程序后,从 zip 文件中提取整个文件夹,并将其存储在 SharePoint 服务器的…

作者头像 李华
网站建设 2026/4/12 23:16:08

Dify镜像部署后的监控与运维策略建议

Dify镜像部署后的监控与运维策略建议 在企业加速拥抱大模型的今天,越来越多团队开始基于Dify构建智能客服、知识库问答、自动化报告生成等AI应用。作为一款开源的可视化LLM应用开发平台,Dify通过拖拽式编排和全生命周期管理能力,显著降低了A…

作者头像 李华
网站建设 2026/4/13 14:15:31

Groq LPU 架构解读为什么它把大模型推理“尾延迟”压得这么稳

1. LPU 的核心目标:为推理而生,而不是从训练芯片“改装” Groq 在架构页的定位很直白:Designed for inference. Not adapted for it.(Groq) 它想解决的不是“训练吞吐最大化”,而是推理里最难受的两点: 单请求&#…

作者头像 李华