C++从零开始的打怪升级之路(day46)

2024-03-08 21:04

本文主要是介绍C++从零开始的打怪升级之路(day46),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这是关于一个普通双非本科大一学生的C++的学习记录贴

在此前,我学了一点点C语言还有简单的数据结构,如果有小伙伴想和我一起学习的,可以私信我交流分享学习资料

那么开启正题

今天分享的是关于二叉树的题目

1.从前序与中序遍历序列构造二叉树

105. 从前序与中序遍历序列构造二叉树

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

这道题是很经典的二叉树题目,原理就是利用前序插入数据,中序判断左右子树,构建二叉树,利用了前序和中序的性质,同时也利用了递归的思想

class Solution {
public:TreeNode* _buildTree(vector<int>&preorder, vector<int>& inorder, int inbegin, int inend, int& prei){//判断是否需要构建if(inbegin > inend)return nullptr;//构建头结点TreeNode* root = new TreeNode(preorder[prei]);//查找分隔点int rooti = inbegin;while(rooti <= inend){if(inorder[rooti] == preorder[prei])break;else++rooti;}//如果有数据递归链接左右子树if(inbegin <= rooti - 1)root->left = _buildTree(preorder, inorder, inbegin, rooti - 1, ++prei);if(rooti + 1 <= inend)root->right = _buildTree(preorder, inorder, rooti + 1, inend, ++prei);return root;}TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {int inbegin = 0, inend = inorder.size() - 1, prei = 0;//原函数无法直接完成递归,借助子函数完成TreeNode* ret = _buildTree(preorder, inorder, inbegin, inend, prei);return ret;}
};

这是ac代码 

2.从中序与后序遍历蓄力构造二叉树

106. 从中序与后序遍历序列构造二叉树

给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 

这题与上面的题是姊妹题,仅有细微的差别,这里直接给代码

class Solution {
public:TreeNode* _buildTree(vector<int>& inorder, vector<int>& postorder, int inbegin, int inend, int& posti){//判断是否需要构建if(inbegin > inend)return nullptr;//创建头结点TreeNode* root = new TreeNode(postorder[posti]);//查找分隔点int rooti = inbegin;while(rooti <= inend){if(inorder[rooti] == postorder[posti])break;else++rooti;}//递归创建左右子树并链接if(rooti + 1 <= inend)root->right = _buildTree(inorder, postorder, rooti + 1, inend, --posti);if(inbegin <= rooti - 1)root->left = _buildTree(inorder, postorder, inbegin, rooti - 1, --posti);return root;}TreeNode* buildTree(vector<int>& inorder, vector<int>& postorder){   int inbegin = 0, inend = inorder.size() - 1, posti = postorder.size() - 1;TreeNode* ret = _buildTree(inorder, postorder, inbegin, inend, posti);return ret;}
};

这是ac代码 

3.二叉树的前序遍历

144. 二叉树的前序遍历

给你二叉树的根节点 root ,返回它节点值的 前序 遍历。

二叉树的三种遍历,在前面的学习都已经很熟悉了,这里我们着重要掌握的是非递归的方法,也是以后面试的一个小难题

这里先给出简单的递归解决办法

class Solution {
public:void _preorderTraversal(TreeNode* root, vector<int>& v){if(root == nullptr)return;v.push_back(root->val);_preorderTraversal(root->left, v);_preorderTraversal(root->right, v);}vector<int> preorderTraversal(TreeNode* root) {   vector<int> v;_preorderTraversal(root, v);return v;}
};

那么,非递归方法要如何求解呢,首先我们要明白,非递归方法是利用迭代来实现的,但其原理还是根据递归的办法来解决的,实际运用中,递归由于要不停的创建函数栈帧,有效率损耗(实际上优化后,时间差的并不多),以及可能导致栈溢出,所以我们才需要掌握非递归办法

非递归办法要创建一个stack,push进vector同时,入栈,在栈内模拟递归

class Solution {
public:vector<int> preorderTraversal(TreeNode* root) {vector<int> v;stack<TreeNode*> st;TreeNode* cur = root;while(cur || !st.empty()){while(cur){st.push(cur);v.push_back(cur->val);cur = cur->left;}TreeNode* top = st.top();st.pop();cur = top->right;}return v;}
};

这是ac代码

4.二叉树的中序遍历

94. 二叉树的中序遍历

和前序遍历如出一辙,这里直接给出递归代码,以及非递归代码

a.递归方法

class Solution {
public:void _inorderTraversal(TreeNode* root, vector<int>& v) {if(root == nullptr)return;_inorderTraversal(root->left, v);v.push_back(root->val);_inorderTraversal(root->right, v);}vector<int> inorderTraversal(TreeNode* root) {vector<int> v;_inorderTraversal(root, v);return v;}
};

b.非递归方法

class Solution {
public:vector<int> inorderTraversal(TreeNode* root) {vector<int> v;stack<TreeNode*> st;TreeNode* cur = root;while(cur || !st.empty()){while(cur){st.push(cur);cur = cur->left;}TreeNode* top = st.top();st.pop();v.push_back(top->val);cur = top->right;}return v;}
};

5.二叉树的后序遍历

和上面的题相似,但是后序遍历要更复杂一点,因为要最后读取根,所以我们要记录读取的值,判断是否遍历完了子树再对根进行读取

class Solution {
public:void _postorderTraversal(TreeNode* root, vector<int>& v){if(!root)return;_postorderTraversal(root->left, v);_postorderTraversal(root->right, v);v.push_back(root->val);}vector<int> postorderTraversal(TreeNode* root) {vector<int> v;_postorderTraversal(root, v);return v;}
};
class Solution {
public:vector<int> postorderTraversal(TreeNode* root) {vector<int> v;stack<TreeNode*> st;TreeNode* cur = root;TreeNode* last = nullptr;while(cur || !st.empty()){while(cur){st.push(cur);cur = cur->left;}TreeNode* top = st.top();if(top->right == nullptr || last == top->right){st.pop();last = top;v.push_back(top->val);}else{cur = top->right;}} return v;}
};

这是ac代码

新手写博客,有不对的位置希望大佬们能够指出,也谢谢大家能看到这里,让我们一起学习进步吧!

这篇关于C++从零开始的打怪升级之路(day46)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

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

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

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

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

Ubuntu如何升级Python版本

《Ubuntu如何升级Python版本》Ubuntu22.04Docker中,安装Python3.11后,使用update-alternatives设置为默认版本,最后用python3-V验证... 目China编程录问题描述前提环境解决方法总结问题描述Ubuntu22.04系统自带python3.10,想升级

解决升级JDK报错:module java.base does not“opens java.lang.reflect“to unnamed module问题

《解决升级JDK报错:modulejava.basedoesnot“opensjava.lang.reflect“tounnamedmodule问题》SpringBoot启动错误源于Jav... 目录问题描述原因分析解决方案总结问题描述启动sprintboot时报以下错误原因分析编程异js常是由Ja

深入解析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 算法核心思想归并排序是一种高效的排序方式,需要用