《数据结构风云》:二叉树遍历的底层思维>递归与迭代的双重视角

《数据结构风云》:二叉树遍历的底层思维>递归与迭代的双重视角


在这里插入图片描述


🔥@晨非辰Tong: 个人主页
👀专栏:《C语言》《数据结构与算法入门指南》
💪学习阶段:C语言、数据结构与算法初学者
⏳“人理解迭代,神理解递归。”


文章目录

引言

二叉树遍历是理解树结构操作的基础,也是算法设计核心环节。前、中与后序遍历,以不同顺序访问节点,体现了递归与迭代思想的精髓。掌握转换规律,不仅能提升对数据结构本质的理解,更能为解决复杂算法问题奠定基础。本文将通过经典题目,解析如何还原二叉树,深入剖析遍历背后的逻辑与实现方法。
获取原码》点我《!!!

知识点前瞻

在这里插入图片描述

一、不一样的前序遍历

144. 二叉树的前序遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

首先看平台给出的接口实现框架——>int* preorderTraversal(struct TreeNode*root, int* returnSize),这时候再看输出示例:返回的是一个数组,那么框架应该就是来返回数组的。对于returnSize猜测是目标树的节点个数,但是输出中没有给出,那么要自己去实现求个数接口
然后根据求出的节点个数去开辟数组空间(因为Note: The returned array must be malloced, assume caller calls free().)。
最后,就要实现前序遍历,但这个前序遍历与之前实现不太一样:不需要打印出节点的数值,只需要将数值存储在要返回的数组中。

复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现

/** * Definition for a binary tree node. * struct TreeNode { * int val; * struct TreeNode *left; * struct TreeNode *right; * }; *//** * Note: The returned array must be malloced, assume caller calls free(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//前序遍历接口voidPreOrder(structTreeNode* root,int* arr,int* pi){//空树,直接返回if(root ==NULL){return;}//将非空节点的值存储在要返回的数组 arr[(*pi)++]= root->val;//遍历子树PreOrder(root->left, arr, pi);PreOrder(root->right, arr, pi);}//返回数组接口//returnSize:树的节点个数,输入为给出,代表要自己计算int*preorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//前序遍历int i =0;PreOrder(root, arr,&i);return arr;}
在这里插入图片描述

3.2 注意要点

  1. 变量i的定义、传参:程序中数据存放在数组中需要下标i,并没有在前序遍历接口中创建或者创建全局变量,而是通过传参(会导致i重复初始化或者累加,前面说过)。但是传的是地址,如果只传数值的话,在后续递归调用函数,这个i不会随着元素的增加改变.

二、不一样的中序遍历

94. 二叉树的中序遍历

1.要求描述:

在这里插入图片描述

2.实现示例

在这里插入图片描述

3.算法思路:

整体思路与上面的前序遍历大致相同,只需要将 arr[(*pi)++] = root->val;放在PreOrder(root->left, arr, pi); PreOrder(root->right, arr, pi);中间即可。实现左根右。
复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现:

/** * Definition for a binary tree node. * struct TreeNode { * int val; * struct TreeNode *left; * struct TreeNode *right; * }; *//** * Note: The returned array must be malloced, assume caller calls free(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//中序遍历voidInOrder(structTreeNode* root,int* arr,int* pi){//树为空if(root ==NULL){return;}//树不为空//遍历左子树InOrder(root->left, arr, pi); arr[(*pi)++]= root->val;InOrder(root->right, arr, pi);}int*inorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//中序遍历int i =0;InOrder(root, arr,&i);return arr;}

三、不一样的后序遍历

145. 二叉树的后序遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

整体思路与上面的中序遍历大致相同,只需要将 arr[(*pi)++] = root->val;放在PreOrder(root->left, arr, pi); PreOrder(root->right, arr, pi);后面即可。实现左右根。
复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现:

/** * Definition for a binary tree node. * struct TreeNode { * int val; * struct TreeNode *left; * struct TreeNode *right; * }; *//** * Note: The returned array must be malloced, assume caller calls free(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//后序遍历voidPostOrder(structTreeNode* root,int* arr,int* pi){//树为空if(root ==NULL){return;}//树不为空//遍历左子树PostOrder(root->left, arr, pi);PostOrder(root->right, arr, pi); arr[(*pi)++]= root->val;}int*postorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//后序遍历int i =0;PostOrder(root, arr,&i);return arr;}
在这里插入图片描述

四、二叉树遍历

TSINGK110 二叉树遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

–牛客网平台全部代码都需要我们自己去实现,比较麻烦。 首先,根据描述:我们需要将用户输入的前序遍历完成的字符串存放在数组中,再根据数组来重现树的结构——>自定义创建树函数。创建树就需要知道树节点的结构,再申请节点——>定义树节点的结构、自定义创建节点函数。
上面这些函数的实现,我们前面都操作过。 然后,就是要中序遍历,这个我们也实现过。

复杂度:

  • 时间复杂度:O(N) ;
  • 空间复杂度:O(N);

3.1具体代码实现

#include<stdio.h>#include<stdlib.h>//定义二叉树的结构typedefstructBinaryTreeNode{char data;structBinaryTreeNode* left;structBinaryTreeNode* right;}BTNode;//根据字符创建节点 BTNode*buyNode(char ch){ BTNode* newnode =(BTNode*)malloc(sizeof(BTNode)); newnode->data = ch; newnode->left = newnode->right =NULL;return newnode;}//创建树 BTNode*creatTree(char* arr,int* pi){//如果是空节点,返回空if(arr[*pi]=='#'){(*pi)++;returnNULL;}//创建新节点--根左右 BTNode* root =buyNode(arr[(*pi)++]); root->left =creatTree(arr, pi); root->right =creatTree(arr, pi);return root;//最终返回指向根节点的指针}//中序遍历voidInOrder(BTNode* root){if(root ==NULL){return;}//左右根InOrder(root->left);printf("%c ", root->data);InOrder(root->right);}intmain(){//读取用户输入的字符串char arr[100];scanf("%s", arr);//根据字符串数组创建二叉树(前序遍历的)int i =0;//接收创建树的根节点 BTNode* root =creatTree(arr,&i);//中序遍历InOrder(root);return0;}

总结

🍓 我是晨非辰Tong!若这篇技术干货帮你打通了学习中的卡点: 👀 【关注】跟我一起深耕技术领域,从基础到进阶,见证每一次成长 ❤️ 【点赞】让优质内容被更多人看见,让知识传递更有力量 ⭐ 【收藏】把核心知识点、实战技巧存好,需要时直接查、随时用 💬 【评论】分享你的经验或疑问(比如曾踩过的技术坑?),一起交流避坑 🗳️ 【投票】用你的选择助力社区内容方向,告诉大家哪个技术点最该重点拆解 技术之路难免有困惑,但同行的人会让前进更有方向~愿我们都能在自己专注的领域里,一步步靠近心中的技术目标! 

二叉树遍历序列的互推,核心在于把握“后序定根,中序分左右”的规律。掌握这一原理,不仅能解决序列重建问题,更能深化对递归和树结构的理解,为学习更复杂的数据结构奠定基础。

Read more

43-dify案例分享-MCP-Server让工作流秒变第三方可调用服务

43-dify案例分享-MCP-Server让工作流秒变第三方可调用服务

1.前言 之前我们为大家介绍过MCP SSE插件,它能够支持MCP-server在Dify平台上的调用,从而帮助Dify与第三方平台提供的MCP-server进行无缝对接。有些小伙伴提出了疑问:既然Dify可以通过MCP SSE插件调用其他平台的MCP-server,那么Dify的工作流或Chatflow是否也能发布为MCP-server,供其他支持MCP client的工具使用呢?今天,我们将为大家介绍一款Dify插件——mcp-server,它能够实现这一功能,即将Dify的工作流或Chatflow发布为MCP-server,供其他第三方工具调用。 插件名字叫做MCP-server,我们在dify插件市场可以找到这个工具 Mcp-server 是一个由 Dify 社区贡献的 Extension 类型插件。安装后,你可以把任何 Dify 应用转变成符合 MCP 标准的 Server Endpoint,供外部 MCP 客户端直接访问。它的主要功能包括: * **暴露为 MCP 工具:**将 Dify 应用抽象为单一 MCP 工具,供外部 MCP 客户端(如

By Ne0inhk
【大模型系列篇】大模型基建工程:基于 FastAPI 自动构建 SSE MCP 服务器

【大模型系列篇】大模型基建工程:基于 FastAPI 自动构建 SSE MCP 服务器

今天我们将使用FastAPI来构建 MCP 服务器,Anthropic 推出的这个MCP 协议,目的是让 AI 代理和你的应用程序之间的对话变得更顺畅、更清晰。FastAPI 基于 Starlette 和 Uvicorn,采用异步编程模型,可轻松处理高并发请求,尤其适合 MCP 场景下大模型与外部系统的实时交互需求,其性能接近 Node.js 和 Go,在数据库查询、文件操作等 I/O 密集型任务中表现卓越。 开始今天的正题前,我们来回顾下相关的知识内容: 《高性能Python Web服务部署架构解析》、《使用Python开发MCP Server及Inspector工具调试》、《构建智能体MCP客户端:完成大模型与MCP服务端能力集成与最小闭环验证》   FastAPI基础知识 安装依赖 pip install uvicorn, fastapi FastAPI服务代码示例  from fastapi import FastAPI app

By Ne0inhk
【MCP】详细了解MCP协议:和function call的区别何在?如何使用MCP?

【MCP】详细了解MCP协议:和function call的区别何在?如何使用MCP?

本文介绍了MCP大模型上下文协议的的概念,并对比了MCP协议和function call的区别,同时用python sdk为例介绍了mcp的使用方式。 1. 什么是MCP? 官网:https://modelcontextprotocol.io/introduction 2025年,Anthropic提出了MCP协议。MCP全称为Model Context Protocol,翻译过来是大模型上下文协议。这个协议的主要为AI大模型和外部工具(比如让AI去查询信息,或者让AI操作本地文件)之间的交互提供了一个统一的处理协议。我们常用的USB TypeC接口(USB-C)统一了USB接口的样式,MCP协议就好比AI大模型中的USB-C,统一了大模型与工具的对接方式。 MCP协议采用了C/S架构,也就是服务端、客户端架构,能支持在客户端设备上调用远程Server提供的服务,同时也支持stdio流式传输模式,也就是在客户端本地启动mcp服务端。只需要在配置文件中新增MCP服务端,就能用上这个MCP服务器提供的各种工具,大大提高了大模型使用外部工具的便捷性。 MCP是开源协议,能让所有A

By Ne0inhk
超详细图文教程:用vscode+copilot(代理模式)便捷使用mcp+一个范例:用自然语言进行3d建模

超详细图文教程:用vscode+copilot(代理模式)便捷使用mcp+一个范例:用自然语言进行3d建模

在vscode使用claude mcp吧! 在vscode更新到最新版本(注意,这是前提)后,内置的copilot可以使用mcp了!!! 关于mcp(Model Context Protocol 模型上下文协议),可以参考我的上一篇文章: MCP个人理解+示例+集成管理+在python中调用示例,给AI大模型装上双手-ZEEKLOG博客 以下是使用教程: 1.点击左下角的齿轮状设置按钮,点击设置 2.在输入面板输入chat.agent.enabled,勾上勾选框 3.点击Ctrl+shift+P,输入reload,点击重新加载窗口,刷新窗口 4.打开copilot后,在右下角将模式改为代理即可。 5.点击工具按钮,开始安装mcp 先去github找到自己想要添加的mcp服务,以blender MCP为例,打开https://github.com/ahujasid/blender-mcp,可以在readme文档里看到详细的安装过程。可以看到,

By Ne0inhk