【题解 | 二叉树】给定二叉树的后序遍历和中序遍历,求层序遍历结果

树的遍历

给定一棵二叉树的后序遍历和中序遍历,请你输出其层序遍历的序列。这里假设键值都是互不相等的正整数。

输入格式:

输入第一行给出一个正整数 N ( ≤ 30 ) N(≤30) N(30),是二叉树中结点的个数。第二行给出其后序遍历序列。第三行给出其中序遍历序列。数字间以空格分隔。

输出格式:

在一行中输出该树的层序遍历的序列。数字间以1个空格分隔,行首尾不得有多余空格。

输入样例:

7
2 3 1 5 7 6 4
1 2 3 4 5 6 7

输出样例:

4 1 6 3 5 7 2

解题

在后序遍历序列中,根节点总是在最后一个位置,而在中序遍历序列中,根节点将序列分为左右两部分,分别对应左子树和右子树。

因此,我们可以利用两个数组的信息,递归构建二叉树,然后再进行层序遍历。

Java

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.Scanner;// 结点类
class TreeNode {int val;			// 值TreeNode left;		// 左孩子TreeNode right;		// 右孩子TreeNode(int x) {val = x;}
}public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int N = scanner.nextInt();int[] postOrder = new int[N];int[] inOrder = new int[N];for (int i = 0; i < N; i++) {postOrder[i] = scanner.nextInt();}for (int i = 0; i < N; i++) {inOrder[i] = scanner.nextInt();}// 根据后序遍历结果和前序遍历结果,构建二叉树TreeNode root = helper(postOrder, inOrder);// 层序遍历List<Integer> ans = levelOrderTraversal(root);// 输出结果for (int i = 0; i < ans.size(); i++) {System.out.print(ans.get(i));if (i < ans.size() - 1) {System.out.print(" ");}}scanner.close();}// 检查是否有某个序列为空private static TreeNode helper(int[] postOrder, int[] inOrder) {if (postOrder == null || inOrder == null || postOrder.length == 0 || inOrder.length == 0) {return null;}return buildTree(postOrder, 0, postOrder.length - 1, inOrder, 0, inOrder.length - 1);}/*** 构建二叉树* @param postOrder 	后序遍历结果* @param postStart 	后序遍历序列的起始位置* @param postEnd 		后序遍历序列的结束位置* @param inOrder 		中序遍历结果* @param inStart 		中序遍历序列的起始位置* @param inEnd 		中序遍历序列的结束位置* @return 构建的二叉树的根节点*/private static TreeNode buildTree(int[] postOrder, int postStart, int postEnd, int[] inOrder, int inStart, int inEnd) {// (0) 终止条件:此时没有结点了,返回空树if (postStart > postEnd || inStart > inEnd) {return null;}// (1) 先找到根结点:后序遍历的最后一个结点int rootVal = postOrder[postEnd];TreeNode root = new TreeNode(rootVal);// (2) 在中序遍历序列中找到根结点的位置int rootIndexInInOrder = 0;for (int i = inStart; i <= inEnd; i++) {if (inOrder[i] == rootVal) {rootIndexInInOrder = i;break;}}// (3) 计算左子树的结点个数int leftTreeSize = rootIndexInInOrder - inStart;// (4) 递归构建左子树和右子树root.left = buildTree(postOrder, postStart, postStart + leftTreeSize - 1, inOrder, inStart, rootIndexInInOrder - 1);root.right = buildTree(postOrder, postStart + leftTreeSize, postEnd - 1, inOrder, rootIndexInInOrder + 1, inEnd);// (5) 返回根结点return root;}// 层序遍历二叉树private static List<Integer> levelOrderTraversal(TreeNode root) {List<Integer> result = new ArrayList<>();if (root == null) {return result;}// 创建队列,用于层序遍历Queue<TreeNode> queue = new LinkedList<>();queue.offer(root);// 循环遍历队列中的结点while (!queue.isEmpty()) {TreeNode node = queue.poll();// 将当前结点的值加入结果列表result.add(node.val);if (node.left != null) {queue.offer(node.left); // 将左子结点加入队列}if (node.right != null) {queue.offer(node.right); // 将右子结点加入队列}}return result;}
}

C++

#include <iostream>
#include <vector>
#include <queue>
using namespace std;// 结点类
class TreeNode {
public:int val;        // 值TreeNode *left;     // 左孩子TreeNode *right;    // 右孩子TreeNode(int x) {val = x;left = nullptr;right = nullptr;}
};// 构建二叉树
TreeNode* buildTree(vector<int>& postOrder, int postStart, int postEnd, vector<int>& inOrder, int inStart, int inEnd) {// 终止条件:此时没有结点了,返回空树if (postStart > postEnd || inStart > inEnd) {return nullptr;}// 先找到根结点:后序遍历的最后一个结点int rootVal = postOrder[postEnd];TreeNode* root = new TreeNode(rootVal);// 在中序遍历序列中找到根结点的位置int rootIndexInInOrder = 0;for (int i = inStart; i <= inEnd; i++) {if (inOrder[i] == rootVal) {rootIndexInInOrder = i;break;}}// 计算左子树的结点个数int leftTreeSize = rootIndexInInOrder - inStart;// 递归构建左子树和右子树root->left = buildTree(postOrder, postStart, postStart + leftTreeSize - 1, inOrder, inStart, rootIndexInInOrder - 1);root->right = buildTree(postOrder, postStart + leftTreeSize, postEnd - 1, inOrder, rootIndexInInOrder + 1, inEnd);// 返回根结点return root;
}// 层序遍历二叉树
vector<int> levelOrderTraversal(TreeNode* root) {vector<int> result;if (root == nullptr) {return result;}// 创建队列,用于层序遍历queue<TreeNode*> q;q.push(root);// 循环遍历队列中的结点while (!q.empty()) {TreeNode* node = q.front();q.pop();// 将当前结点的值加入结果列表result.push_back(node->val);if (node->left != nullptr) {q.push(node->left); // 将左子结点加入队列}if (node->right != nullptr) {q.push(node->right); // 将右子结点加入队列}}return result;
}int main() {int N;cin >> N;vector<int> postOrder(N);vector<int> inOrder(N);for (int i = 0; i < N; i++) {cin >> postOrder[i];}for (int i = 0; i < N; i++) {cin >> inOrder[i];}// 根据后序遍历结果和中序遍历结果,构建二叉树TreeNode* root = buildTree(postOrder, 0, N - 1, inOrder, 0, N - 1);// 层序遍历vector<int> ans = levelOrderTraversal(root);// 输出结果for (int i = 0; i < ans.size(); i++) {cout << ans[i];if (i < ans.size() - 1) {cout << " ";}}return 0;
}

测试链接:L2-006 树的遍历

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/802737.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

grep 和 pgrep

grep 和 pgrep 是两个不同的命令&#xff0c;它们在Linux和类Unix系统中用于搜索文本&#xff0c;但各自的应用场景和功能有所差异。 grep 全称&#xff1a;Global Regular Expression Print 用途&#xff1a;grep 主要用于在文件或者通过管道传递的输出中搜索与指定模式&…

深入探讨udevd:Linux中的设备管理守护进程

Linux操作系统的强大功能之一在于其对硬件的优秀支持。无论是新接入的USB设备还是系统内部的硬件变化&#xff0c;Linux都能够灵活地处理。这得益于udevd—一种设备管理守护进程。在本篇博客中&#xff0c;我们将详细探讨udevd的工作原理、配置和在系统中的重要性。 什么是ude…

深度学习500问——Chapter06: 循环神经网络(RNN)(2)

文章目录 6.4 CNN和RNN的区别 6.5 RNNs与FNNs有什么区别 6.6 RNNs训练和传统ANN训练异同点 6.7 为什么RNN训练的时候Loss波动很大 6.8 标准RNN前向输出流程 6.9 BPTT算法推导 6.9 RNN中为什么会出现梯度消失 6.10 如何解决RNN中的梯度消失问题 6.4 CNN和RNN的区别 类别特点描述…

博客系统实现

一.准备工作 1.创建项目&#xff0c;把前端写好的博客静态页面拷贝到webapp目录中 2.引入依赖&#xff0c;这里主要用到servlet&#xff0c;mysql5.1.47&#xff0c;jacson2.15.0 3.找到右上角的edit configurations->smartTomcat->进行配置 4.数据库设计&#xff1a…

2024年4月9号PMP每日三题含答案

2024年4月9号PMP每日三题含答案 1.在执行一个潜艇现代化项目期间&#xff0c;客户要求安装新的潜望镜。项目经理必须怎么做&#xff1f; A.检查可行性&#xff0c;准备预算&#xff0c;并获得变更请求批准 B.执行实施整体变更控制过程&#xff0c;获得预算批准&#xff0c;并执…

Java8 函数式编程 @FunctionInterface使用示例

FunctionInterface 是一个注解&#xff0c;用在接口上面。 接口内部只能有一个方法。 作用&#xff1a; 将函数作为参数传入其它方法。 背景 看如下代码&#xff0c; 发现send1 send2 send3 方法都有共同的代码用于获取参数&#xff0c;唯一不同的是消息发送逻辑不一样。 那…

吴恩达深度学习 (week1,2)

文章目录 1、神经网络监督学习2、深度学习兴起原因3、深度学习二元分类4、深度学习Logistic 回归5、Logistic 回归损失函数6、深度学习梯度下降法7、深度学习向量法8、Python 中的广播9、上述学习总结10、大作业实现:rocket::rocket:&#xff08;1&#xff09;训练初始数据&…

Matlab进阶绘图第49期—气泡堆叠图

气泡堆叠图是堆叠图与气泡图的组合—在堆叠图每根柱子上方添加大小不同的气泡&#xff0c;用于表示另外一个数据变量&#xff08;如每根柱子各组分的平均值&#xff09;的大小。 本文利用自己制作的BarBubble工具&#xff0c;进行气泡堆叠图的绘制&#xff0c;先来看一下成品效…

代码随想录算法训练营第三十五天| 860.柠檬水找零,406.根据身高重建队列,452. 用最少数量的箭引爆气球

题目与题解 860.柠檬水找零 题目链接&#xff1a;860.柠檬水找零 代码随想录题解&#xff1a;​​​​​​​860.柠檬水找零 视频讲解&#xff1a;贪心算法&#xff0c;看上去复杂&#xff0c;其实逻辑都是固定的&#xff01;LeetCode&#xff1a;860.柠檬水找零_哔哩哔哩_bil…

GET请求和POST请求的区别

最为常见的客户端传递参数方式有两种&#xff1a; GET 请求和 POST 请求底层都是基于 TCP/IP 协议实现的&#xff0c;使用二者中的任意一个&#xff0c;都可以实现客户端和服务器端的双向交互。 浏览器地址栏直接输入&#xff1a;一定是GET请求&#xff1b; 超链接&#xff1a;…

从数据采集到可视化展示Node-Red二次开发4G模块

环境监测正逐步迈入数字化、智能化时代。Node-Red作为一种开源流式编程工具&#xff0c;以其强大的数据处理能力和设备集成便捷性&#xff0c;在构建环境监测数据站中发挥着至关重要的作用。钡铼技术支持Node-Red编程开发&#xff0c;支持BLIoTLink软网关和自定义开发非标协议。…

6. TypeScript的枚举类型

在TypeScript中&#xff0c;枚举&#xff08;Enum&#xff09;是一种特殊的数据类型&#xff0c;它允许为一组数值赋予友好的名字。枚举类型在处理一组相关常量时非常有用&#xff0c;比如状态码、方向、月份等。本文将从多个方面深入探讨TypeScript中枚举类型的使用&#xff0…

为什么 MySQL 采用 B+ 树作为索引?

资料来源 : 小林coding 小林官方网站 : 小林coding (xiaolincoding.com) 「为什么 MySQL 采用 B 树作为索引&#xff1f;」这句话&#xff0c;是不是在面试时经常出现。 要解释这个问题&#xff0c;其实不单单要从数据结构的角度出发&#xff0c;还要考虑磁盘 I/O 操作次数&am…

【javaWeb】EL与JSTL

目录 EL表达式&#xff08;Expression Language&#xff09;EL和JSP脚本的区别语法EL操作符EL运算符注意 EL功能EL访问作用域隐式对象示例1&#xff1a;获得应用上下文示例2&#xff1a;获取Cookie对象 JSTL(JavaServerPages Standard Tag Library)JSTL使用步骤JSTL标签分类使用…

【免费题库】华为OD机试 - 单词重量(Java JS Python C C++)

须知 哈喽,本题库完全免费,收费是为了防止被爬,大家订阅专栏后可以私信联系退款。感谢支持 文章目录 须知题目描述输入描述输出描述用例解题思路:Java代码:JS代码:Python代码:C代码:C++代码:题目描述 每个句子由多个单词组成,句子中的每个单词的长度都可能不一样,…

【六 (3)机器学习-机器学习建模步骤/kaggle房价回归实战】

目录 文章导航一、确定问题和目标&#xff1a;1、业务需求分析&#xff1a;2、问题定义&#xff1a;3、目标设定&#xff1a;4、数据可行性评估&#xff1a;5、资源评估&#xff1a;6、风险评估&#xff1a; 二、数据收集&#xff1a;1、明确数据需求2、选择数据来源3、考虑数据…

SpringCloud Alibaba Seata 处理分布式事务

一、前言 接下来是开展一系列的 SpringCloud 的学习之旅&#xff0c;从传统的模块之间调用&#xff0c;一步步的升级为 SpringCloud 模块之间的调用&#xff0c;此篇文章为第十八篇&#xff0c;即使用 Seata 处理分布式事务。 二、分布式事务问题 当单体应用被拆分成微服务应用…

实现虚拟机Ubuntu与主机Windows之间的复制粘贴

实现步骤 在虚拟机Ubuntu界面&#xff0c;CtrlAltT&#xff0c;新建一个终端 # 卸载已有的工具 sudo apt-get autoremove open-vm-tools# 安装工具open-vm-tools sudo apt-get install open-vm-tools# 安装open-vm-tools-desktop sudo apt-get install open-vm-tools-desk…

【超简单】基于PaddleSpeech搭建个人语音听写服务

一、【超简单】之基于PaddleSpeech搭建个人语音听写服务 1.需求分析 亲们,你们要写会议纪要嘛?亲们,你们要写会议纪要嘛?亲们,你们要写会议纪要嘛?当您面对成吨的会议录音,着急写会议纪要而不得不愚公移山、人海战术?听的头晕眼花,听的漏洞百出,听的怀疑人生,那么你…

在Android中使用MediaPlayer播放音频和视频

在播放音频中SoundPool是个很好用的类&#xff0c;但是SoundPool只能播放音频&#xff0c;且通常只用来播放较短的音频&#xff0c;这就需要另外的类来实现视频和长音频的播放&#xff0c;那就是MediaPlay 实现MediaPlay的基本步骤是 创建MediaPlay对象调用setDataSource对象…