【C++】每日一题 105 从前序和中序序列构造二叉树

2024-03-17 09:52

本文主要是介绍【C++】每日一题 105 从前序和中序序列构造二叉树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

#include <iostream>
#include <vector>
#include <unordered_map>struct TreeNode {int val;TreeNode* left;TreeNode* right;TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};TreeNode* buildTreeHelper(std::vector<int>& preorder, std::vector<int>& inorder, int preStart, int inStart, int inEnd, std::unordered_map<int, int>& indexMap) {if (preStart >= preorder.size() || inStart > inEnd) {return nullptr;}int rootVal = preorder[preStart];TreeNode* root = new TreeNode(rootVal);int inIndex = indexMap[rootVal];root->left = buildTreeHelper(preorder, inorder, preStart + 1, inStart, inIndex - 1, indexMap);root->right = buildTreeHelper(preorder, inorder, preStart + inIndex - inStart + 1, inIndex + 1, inEnd, indexMap);return root;
}TreeNode* buildTree(std::vector<int>& preorder, std::vector<int>& inorder) {if (preorder.empty() || inorder.empty() || preorder.size() != inorder.size()) {return nullptr;}std::unordered_map<int, int> indexMap;for (int i = 0; i < inorder.size(); ++i) {indexMap[inorder[i]] = i;}return buildTreeHelper(preorder, inorder, 0, 0, inorder.size() - 1, indexMap);
}void printInorder(TreeNode* root) {if (root) {printInorder(root->left);std::cout << root->val << " ";printInorder(root->right);}
}int main() {std::vector<int> preorder = {3, 9, 20, 15, 7};std::vector<int> inorder = {9, 3, 15, 20, 7};TreeNode* root = buildTree(preorder, inorder);std::cout << "Inorder traversal of the constructed tree: ";printInorder(root);std::cout << std::endl;return 0;
}

当给定先序遍历 preorder 和中序遍历 inorder 数组时,我们可以通过递归的方式构造二叉树。详细的构造过程如下:

确定根节点:

先序遍历的第一个元素必定是当前子树的根节点的值。
在中序遍历数组中找到根节点的位置,根节点左边的元素属于左子树,右边的元素属于右子树。

递归构造左子树:

根据中序遍历的结果,左子树的元素位于根节点左边,右子树的元素位于根节点右边。
在先序遍历中,左子树的元素紧随在根节点后面,因此可以使用递归构造出左子树。

递归构造右子树:

同理,右子树的构造也可以通过递归完成。
重复以上步骤:

对左子树和右子树分别进行递归构造,直到所有节点都被处理。
在实现代码中,我们使用 buildTreeHelper 函数来辅助递归构造二叉树。该函数接受先序遍历数组、中序遍历数组、当前子树在先序遍历中的起始位置、当前子树在中序遍历中的起始位置和结束位置以及存储中序遍历元素索引的哈希表。根据这些参数,函数递归构造二叉树并返回根节点。

最后,在 main 函数中,调用 buildTree 函数传入先序遍历数组和中序遍历数组,构造出二叉树的根节点,并输出该二叉树的中序遍历结果作为验证。整个构造过程就是根据先序遍历和中序遍历的特性,逐步确定每个节点的位置,然后递归构造左右子树,最终构建整棵二叉树。

这篇关于【C++】每日一题 105 从前序和中序序列构造二叉树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/818604

相关文章

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp