剑指offer面试题6 重建二叉树

考察点

知识点

链表中每个结点最少有1个指针,最多2(双重链表),前后结点是一对一的关系,而树中每个结点指针数量可以更多一些,也就是说树中的结点存储着更多其它结点的信息,前后结点是一对多的关系(其中靠前的结点有个专门的术语叫父结点,靠后的结点都是孩子结点,没有孩子结点的叫叶子结点,没有父结点的叫根结点)。这俩个数据结构需要解决的问题不一样。依据每个结点包含的指针数量的不同对树做了一些分类,比如每个结点最多包含俩个指针的树叫做二叉树(这里又给这些结点起了个名字二叉树中偏左的叫左结点偏右的叫右结点)
同其它类型的数据结构有一样,树的操作也无外乎创建结点,删除结点,遍历结点,其中创建和删除的过程又都需要用到遍历,遍历一颗普通的树需要O(n),遍历的方式包括前序遍历(根结点-左结点-右结点),中序遍历(左结点-根结点-右结点),后序遍历(左结点-右结点-根结点),遍历的代码实现包括递归和迭代俩种方式,下面代码中有实现。其实递归就是用了程序自己的栈,迭代是用了我们自己创建的栈。
数据结构的目的肯定是为了更高效的使用数据,基于树的这种结构,如果把比父结点小的数据都存左边,比父结点大的数据都存右边,这样查找数据的时候就有点类似于二分查找了,时间复杂度降到O(logn),这样的树被称作搜索树

题目

分析
本题目输入的一个树的前序和中序遍历序列,要求把这颗树重建起来。这里需要了解到程序中有个特别重要的思维叫递归,递归是分层的,每一层的问题都是一样的,只不过问题的规模不一样,面对这类复杂问题你的思考方式一定是2点:1.问题该如何定义?第n层该如何解决?如果第n层的问题已经解决好了,那么就继续用第n层的方式去解决第n-1层;2.第0层的问题是怎么样的该怎么解决。套到这个题目中问题显然就是用一个前序序列和一个中序序列构建树,(1思维展现)第n层问题怎么解决?前序遍历第一个元素就是根结点,而中序遍历左根右如果能找到根结点在中序序列中的位置,那么就能确定该根结点左边的所有结点都是左子树的中序序列,右边的所有结点都是右子树的中序序列,结点数目知道了以后再返回前序遍历找出左子树的前序序列和右子树的前序序列,这样一来就可以继续用这个方法去分别构建左子树和右子树。(2思维展现)第0层肯定是问题规模最小的那一层,本题目中即只有一个结点的时候,这种情况直接解决本层问题就可以了。

public class Node{int val;Node leftChild;Node rightChild;public Node(int data) {this.val = data;this.leftChild = null;this.rightChild = null;}
}
import java.util.Deque;
import java.util.LinkedList;public class BinaryTree {Node root;public BinaryTree() {this.root = null;}public void insertTree(int val) {if (this.root == null) {Node root = new Node(val);this.root = root;} else {insertChildTree(this.root,val);}}public void insertChildTree(Node node,int val) {if (node != null && val < node.val) {if (node.leftChild == null) {node.leftChild = new Node(val);} else {insertChildTree(node.leftChild,val);}}if (node != null && val > node.val) {if (node.rightChild == null) {node.rightChild = new Node(val);} else {insertChildTree(node.rightChild,val);}}}public Node getRoot() {return this.root;}public void prePrint(Node node) {if (node == null) {return;}System.out.print(node.val + " ");prePrint(node.leftChild);prePrint(node.rightChild);}public void midPrint(Node node) {if (node == null) {return;}midPrint(node.leftChild);System.out.print(node.val + " ");midPrint(node.rightChild);}public void afterPrint(Node node) {if (node == null) {return;}afterPrint(node.leftChild);afterPrint(node.rightChild);System.out.print(node.val + " ");}public void prePrintIteration(Node root) {Deque<Node> stack = new LinkedList<>();Node node = root;while (stack.size() > 0 || node != null) {while (node != null) {System.out.print(node.val + " ");stack.push(node);node = node.leftChild;}if (stack.size() > 0) {node = stack.peek().rightChild;stack.pop();}}}public void midPrintIteration(Node root) {Deque<Node> stack = new LinkedList<>();Node node = root;while (stack.size() > 0 || node != null) {while (node != null) {stack.push(node);node = node.leftChild;}//执行到此说明左子树都已入栈if (stack.size() > 0) {Node parent = stack.peek();//父结点System.out.print(parent.val + " ");node = parent.rightChild;stack.pop();}}}public void afterPrintIteration(Node root) {Deque<Node> stack = new LinkedList<>();Node pre = null;Node node = root;while (stack.size() > 0 || node != null) {while (node != null) {stack.push(node);node = node.leftChild;}Node parent = stack.peek();if (parent.rightChild == pre || parent.rightChild == null) {System.out.print(parent.val + " ");pre = parent;stack.pop();} else {node = parent.rightChild;}}}public void constructPrint(int[] pre,int[] mid) {if (pre.length != mid.length || pre.length <= 0) {throw new Error("params error");}int len = pre.length;Node root = constructTree(pre,0,len-1,mid,0,len-1);prePrint(root);System.out.println();midPrint(root);System.out.println();}public Node constructTree(int[] pre,int preStart,int preEnd,int[] mid,int midStart,int midEnd) {int rootVal = pre[preStart];Node root = new Node(rootVal);root.leftChild = null;root.rightChild = null;if (preStart == preEnd && midStart == midEnd) {if (pre[preStart] != mid[midStart]) {throw new Error("input error");} else {return root;}}int leftMidEnd = midStart;for (int i = midStart;i <= midEnd; i++) {if (rootVal != mid[i]) {leftMidEnd ++;} else {break;}}//根结点在中序序列中不存在if (leftMidEnd > midEnd) {throw new Error("input error");}int leftLength = leftMidEnd - midStart;if (leftLength > 0) {root.leftChild = constructTree(pre,preStart + 1,preStart + leftLength,mid,midStart,leftMidEnd - 1);}if (leftLength < midEnd - midStart) {root.rightChild = constructTree(pre,preStart + leftLength + 1,preEnd,mid,leftMidEnd + 1,midEnd);}return root;}
}
public class Six {public static void main(String[] args) {//树的遍历BinaryTree tree = new BinaryTree();tree.insertTree(10);tree.insertTree(6);tree.insertTree(14);tree.insertTree(4);tree.insertTree(8);tree.insertTree(12);tree.insertTree(16);tree.prePrint(tree.getRoot());System.out.println();tree.prePrintIteration(tree.getRoot());System.out.println();tree.midPrint(tree.getRoot());System.out.println();tree.midPrintIteration(tree.getRoot());System.out.println();tree.afterPrint(tree.getRoot());System.out.println();tree.afterPrintIteration(tree.getRoot());System.out.println();int[] pre = {1,2,4,7,3,5,6,8};int[] mid = {4,7,2,1,5,3,8,6};//本题目重建二叉树BinaryTree treeBuild = new BinaryTree();treeBuild.constructPrint(pre,mid);}
}

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

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

相关文章

[力扣 Hot100]Day8 无重复字符的最长子串

题目描述 给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长子串 的长度。 出处 思路 使用双指针构成一个滑动窗口&#xff0c;维护窗口内的字符不重复&#xff0c;逐步右移右指针&#xff0c;当右指针所指元素已经在窗口内时&#xff0c;左指针右移到和右指…

1308. 全排列的结果-深度优先搜索-DFS

代码&#xff1a; #include <iostream> using namespace std; int n; int a[10]; bool r[10]; void print(int k) {for(int i1; i<k; i){cout<<a[i]<<" ";}cout<<endl; } void fun(int k) {for(int i1; i<n; i){if(r[i] false){a[k…

第一讲_HarmonyOS应用开发环境准备

HarmonyOS应用开发环境准备 1. 知识储备2. 环境搭建2.1 安装node.js2.2 配置node.js2.3 安装命令行工具2.4 安装DevEco Studio2.5 配置DevEco Studio 1. 知识储备 HarmonyOS提供了一套UI开发框架&#xff0c;即方舟开发框架&#xff08;ArkUI框架&#xff09;。方舟开发框架可…

springboot minio 工具类,一站式解决

注意 minio 新版本有9000和9090两个端口&#xff0c;web访问是9000&#xff0c;但走api上传和访问都是9090端口 引入pom <dependency><groupId>io.minio</groupId><artifactId>minio</artifactId><version>8.2.0</version></d…

给WordPress网站增加一个带时间的led广告牌

WordPress 后台》外观》小工具》自定义html》添加到合适位置 其他系统可自行添加合适位置 <style type"text/css">.studytextgzbox {background: #F9F9F9; border: 1px solid #999999;margin: 1px;text-align:center; float: left;line-height: 28px;hei…

Qt中ComboBox的简单使用

1.相关说明 combobox中item的文字、data、图片设置 2.界面绘制 3.相关主要代码 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this); }Widget::~Widget() {delete …

回归预测 | Matlab实现MSADBO-CNN-LSTM基于改进蜣螂算法优化卷积神经网络-长短期记忆神经网络多特征回归预测

回归预测 | Matlab实现MSADBO-CNN-LSTM基于改进蜣螂算法优化卷积神经网络-长短期记忆神经网络多特征回归预测 目录 回归预测 | Matlab实现MSADBO-CNN-LSTM基于改进蜣螂算法优化卷积神经网络-长短期记忆神经网络多特征回归预测预测效果基本描述程序设计参考资料 预测效果 基本描…

从0开始python学习-48.pytest框架之断言

目录 1. 响应进行断言 1.1 在yaml用例中写入断言内容 1.2 封装断言方法 1.3 在执行流程中加入断言判断内容 2. 数据库数据断言 2.1 在yaml用例中写入断言内容 2.2 连接数据库并封装执行sql的方法 2.3 封装后校验方法是否可执行 2.4 使用之前封装的断言方法&#xff0c…

关于VScode的这个ssh的配置的经验

1.首先&#xff0c;我是因为重装了ubantu系统&#xff0c;不得不重新配置ssh 2.第一步&#xff0c;在本机的终端安装ssh插件&#xff1a; &#xff08;1&#xff09; &#xff08;2&#xff09;restart开启这个ssh端口 3.然后&#xff0c;就在vscode里面&#xff0c;安装哪个…

使用CSS计算高度铺满屏幕

前言 今天写项目时出现高度设置百分百却不占满屏幕&#xff0c;第一反应看自己设置的是块级元素还是行级元素。看了几篇博客&#xff0c;发现并不能解决问题。脱离文档流的做法都没考虑&#xff0c;前期模板搭建脱离文档流&#xff0c;后面开发会出现很多问题。 以上图片是我…

FindMy技术与相机结合

FindMy是苹果公司提供的设备追踪服务&#xff0c;用来帮助用户定位丢失的设备。自苹果公司开放Findmy网络之后&#xff0c;FindMy技术便与各种生活设备相结合&#xff0c;比如与相机的结合。 想象一下&#xff0c;你正在外出办事或者旅行时&#xff0c;突然意识到相机丢了&…

SSR服务端渲染解决了什么问题?有做过SSR吗?

面试官&#xff1a;SSR解决了什么问题&#xff1f;有做过SSR吗&#xff1f;你是怎么做的&#xff1f; 一、是什么 Server-Side Rendering 我们称其为SSR&#xff0c;意为服务端渲染 指由服务侧完成页面的 HTML 结构拼接的页面处理技术&#xff0c;发送到浏览器&#xff0c;然…

Unity关于新手引导中实现遮罩镂空效果

在新手引导每一步中实现可以遮掉其他部分而显示当前需要点击的部分&#xff0c;只需要在每一步引导的时候设置对应的镂空区域的RectTransform.效果如下图&#xff1a; 代码&#xff1a; public class SelfMaskSet : MaskableGraphic, ICanvasRaycastFilter {[SerializeField]p…

React的合成事件

合成事件&#xff1a;通过事件委托&#xff0c;利用事件传播机制&#xff0c;当事件传播到document时&#xff0c;再进行分发到对应的组件&#xff0c;从而触发对应所绑定的事件&#xff0c;然后事件开始在组件树DOM中走捕获冒泡流程。 原生事件 —— > React事件 —— >…

嵌入式培训机构四个月实训课程笔记(完整版)-C++和QT编程第四天-C++动态联编和虚函数练习(物联技术666)

链接:https://pan.baidu.com/s/1KayCjn6Vem9YFucS8lpCFg?pwd=1688 提取码:1688 设计一个动物类:动物有一个name成员,另外有三个函数sleep(), eat(), play(); 从动物这个类派生出狗类和猫类,在对应的三个函数中实现输出如下信息的功能: dog(or cat) (name) is sleepping…

互联网技术发展:技术行业的深度思考

随着互联网技术的日新月异&#xff0c;我们身处一个信息爆炸的时代。技术行业作为这场变革的引擎&#xff0c;不断推动着社会的进步。然而&#xff0c;当我们沉浸在这场技术盛宴中时&#xff0c;是否曾停下脚步&#xff0c;思考其背后的深远影响&#xff1f;本文将从一个独特的…

macOS 13(本机)golang程序交叉编译成 ARM架构

## 背景 golang程序&#xff08;JuiceFS&#xff09;需要支持ARM64架构&#xff0c;重新编译&#xff1b; 本地环境&#xff1a;macOS&#xff1a;13 ## 操作 安装交叉编译工具&#xff1a; brew install FiloSottile/musl-cross/musl-cross --with-aarch64 可以在 /usr/l…

C++ 一元运算符重载

一元运算符只对一个操作数进行操作&#xff0c;下面是一元运算符的实例&#xff1a; 递增运算符&#xff08; &#xff09;和递减运算符&#xff08; – &#xff09;一元减运算符&#xff0c;即负号&#xff08; - &#xff09;逻辑非运算符&#xff08; ! &#xff09; 一…

【数据结构】二叉树-堆(top-k问题,堆排序,时间复杂度)

&#x1f308;个人主页&#xff1a;秦jh__https://blog.csdn.net/qinjh_?spm1010.2135.3001.5343&#x1f525; 系列专栏&#xff1a;《数据结构》https://blog.csdn.net/qinjh_/category_12536791.html?spm1001.2014.3001.5482 ​​ 目录 堆排序 第一种 ​编辑 第二种 …

Python兴趣篇_模拟编译中的字典运用

在Python基础篇_字典基本操作中说到&#xff0c;字典可以用于存储编译时需要的信息或参数。 本文就来简单讨论一下模拟编译过程中如何利用字典数据类型&#xff1b; 模仿编译过程中&#xff0c;Python 字典可以用于存储和处理与编译相关的各种信息。以下是 Python 字典在模拟编…