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++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

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 中存储指针类型的对象