2023-05-03:给你一棵 二叉树 的根节点 root ,树中有 n 个节点

每个节点都可以被分配一个从 1 到 n 且互不相同的值

另给你一个长度为 m 的数组 queries

你必须在树上执行 m 个 独立 的查询,其中第 i 个查询你需要执行以下操作:

从树中 移除 以 queries[i] 的值作为根节点的子树

题目所用测试用例保证 queries[i] 不 等于根节点的值。

返回一个长度为 m 的数组 answer ,其中 answer[i] 是执行第 i 个查询后树的高度。

注意:

查询之间是独立的,所以在每个查询执行后,树会回到其 初始 状态。

树的高度是从根到树中某个节点的 最长简单路径中的边数 。

输入:root = [5,8,9,2,1,3,7,4,6], queries = [3,2,4,8]。

输出:[3,2,3,2]。

答案2023-05-03:

大体过程:

1.定义和初始化全局变量

  • 使用常量 MAXN 定义数组大小。

  • 定义用于深度优先搜索的四个数组 dfndeepsizemaxlmaxr 和一个计数器 n,保存每个节点的编号、深度、子树大小、左右子树的最大深度。

2.定义深度优先搜索函数 dfs

  • 用一个计数器 i 记录当前节点的编号,并将其存储到数组 dfn 中。

  • 将当前节点的深度 h 存储到数组 deep 中。

  • 将当前节点的子树大小初始化为 1,存储到数组 size 中。

  • 如果当前节点存在左孩子,则递归调用 dfs 函数,并将当前节点的子树大小加上其左孩子的子树大小。

  • 如果当前节点存在右孩子,则递归调用 dfs 函数,并将当前节点的子树大小加上其右孩子的子树大小。

3.在主函数中创建一棵二叉树 root 和一个查询数组 queries

4.对于每个查询 queries[i],执行以下操作:

  • 计算以 queries[i] 为根节点的子树编号范围,即 dfn[queries[i]]dfn[queries[i]]+size[dfn[queries[i]]]-1

  • 将该范围内所有节点的深度保存到数组 maxl 中,并计算其前缀最大值。

  • 将该范围内所有节点的深度保存到数组 maxr 中,并计算其后缀最大值。

  • 计算左右子树的最大深度,取其中的较大值作为删除子树后树的高度。

  • 将结果保存到答案数组 ans 中。

5.返回答案数组。

注意:在每次查询中,需要重新计算左右子树的最大深度,因为每次查询都会修改树的结构。

时间复杂度:

dfs 函数中,对于每个节点最多访问一次,因此该函数的时间复杂度为 O(n),其中 n 是二叉树的节点数。

treeQueries 函数中,需要处理 $m$ 个查询,对于每个查询需要计算左右子树的最大深度,时间复杂度为 O(n),因此总时间复杂度为 O(mn)。

空间复杂度:

在 C++ 中,数组和变量的空间占用量是固定的,因此空间复杂度主要取决于递归调用时堆栈的空间占用量。由于最坏情况下二叉树可能退化成一个链表,因此堆栈空间的最大使用量为 O(n),其中 n 是二叉树的节点数。

除了堆栈空间之外,还需要使用常量大小的额外空间来存储全局变量和临时变量,因此总空间复杂度为 O(n)。

go完整代码如下:

package main

import (
"fmt"
) type TreeNode struct {
Val int
Left *TreeNode
Right *TreeNode
} const MAXN = 100010 var dfn [MAXN]int
var deep [MAXN]int
var size [MAXN]int
var maxl [MAXN]int
var maxr [MAXN]int
var n int func treeQueries(root *TreeNode, queries []int) []int {
n = 0
dfs(root, 0)
for i := 1; i <= n; i++ {
maxl[i] = max(maxl[i-1], deep[i])
}
maxr[n+1] = 0
for i := n; i >= 1; i-- {
maxr[i] = max(maxr[i+1], deep[i])
}
m := len(queries)
ans := make([]int, m)
for i := 0; i < m; i++ {
leftMax := maxl[dfn[queries[i]]-1]
rightMax := maxr[dfn[queries[i]]+size[dfn[queries[i]]]]
ans[i] = max(leftMax, rightMax)
}
return ans
} func dfs(head *TreeNode, h int) {
i := n + 1
dfn[head.Val] = i
deep[i] = h
size[i] = 1
n = i
if head.Left != nil {
dfs(head.Left, h+1)
size[i] += size[dfn[head.Left.Val]]
}
if head.Right != nil {
dfs(head.Right, h+1)
size[i] += size[dfn[head.Right.Val]]
}
} func max(a, b int) int {
if a > b {
return a
}
return b
} func main() {
root := &TreeNode{
Val: 5,
Left: &TreeNode{
Val: 8,
Left: &TreeNode{
Val: 2,
Left: nil,
Right: nil,
},
Right: &TreeNode{
Val: 9,
Left: nil,
Right: nil,
},
},
Right: &TreeNode{
Val: 3,
Left: &TreeNode{
Val: 1,
Left: nil,
Right: nil,
},
Right: &TreeNode{
Val: 7,
Left: &TreeNode{
Val: 4,
Left: nil,
Right: nil,
},
Right: &TreeNode{
Val: 6,
Left: nil,
Right: nil,
},
},
},
}
queries := []int{3, 2, 4, 8}
ans := treeQueries(root, queries)
fmt.Println("The query results are:", ans)
}

c完整代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h> #define MAXN 100010 struct TreeNode {
int val;
struct TreeNode* left;
struct TreeNode* right;
}; int dfn[MAXN];
int deep[MAXN];
int size[MAXN];
int maxl[MAXN];
int maxr[MAXN];
int n; int max0(int a, int b) {
return a > b ? a : b;
} void dfs(struct TreeNode* head, int h);
int* treeQueries(struct TreeNode* root, int* queries, int queriesSize, int* returnSize); int main() {
struct TreeNode node9 = { 9, NULL, NULL };
struct TreeNode node8 = { 8, NULL, &node9 };
struct TreeNode node2 = { 2, NULL, NULL };
struct TreeNode node4 = { 4, NULL, NULL };
struct TreeNode node1 = { 1, NULL, NULL };
struct TreeNode node6 = { 6, NULL, NULL };
struct TreeNode node7 = { 7, &node4, &node6 };
struct TreeNode node3 = { 3, &node1, &node7 };
struct TreeNode node5 = { 5, &node8, &node3 };
struct TreeNode* root = &node5;
int queries[] = { 3, 2, 4, 8 };
int queriesSize = sizeof(queries) / sizeof(int);
int returnSize = 0;
int* ans = treeQueries(root, queries, queriesSize, &returnSize);
printf("The query results are: [");
for (int i = 0; i < returnSize; i++) {
if (i > 0) {
printf(", ");
}
printf("%d", ans[i]);
}
printf("]\n");
free(ans);
return 0;
} void dfs(struct TreeNode* head, int h) {
int i = ++n;
dfn[head->val] = i;
deep[i] = h;
size[i] = 1;
if (head->left != NULL) {
dfs(head->left, h + 1);
size[i] += size[dfn[head->left->val]];
}
if (head->right != NULL) {
dfs(head->right, h + 1);
size[i] += size[dfn[head->right->val]];
}
} int* treeQueries(struct TreeNode* root, int* queries, int queriesSize, int* returnSize) {
n = 0;
dfs(root, 0);
int i;
for (i = 1; i <= n; i++) {
maxl[i] = max0(maxl[i - 1], deep[i]);
}
maxr[n + 1] = 0;
for (i = n; i >= 1; i--) {
maxr[i] = max0(maxr[i + 1], deep[i]);
}
int* ans = (int*)malloc(queriesSize * sizeof(int));
for (i = 0; i < queriesSize; i++) {
int leftMax = maxl[dfn[queries[i]] - 1];
int rightMax = maxr[dfn[queries[i]] + size[dfn[queries[i]]]];
ans[i] = max0(leftMax, rightMax);
}
*returnSize = queriesSize;
return ans;
}

c++完整代码如下:

#include <iostream>
#include <vector> using namespace std; struct TreeNode {
int val;
TreeNode* left;
TreeNode* right;
TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
}; const int MAXN = 100010;
int dfn[MAXN];
int deep[MAXN];
int size0[MAXN];
int maxl[MAXN];
int maxr[MAXN];
int n; void dfs(TreeNode* head, int h) {
int i = ++n;
dfn[head->val] = i;
deep[i] = h;
size0[i] = 1;
if (head->left != nullptr) {
dfs(head->left, h + 1);
size0[i] += size0[dfn[head->left->val]];
}
if (head->right != nullptr) {
dfs(head->right, h + 1);
size0[i] += size0[dfn[head->right->val]];
}
} vector<int> treeQueries(TreeNode* root, vector<int>& queries) {
n = 0;
dfs(root, 0);
for (int i = 1; i <= n; i++) {
maxl[i] = max(maxl[i - 1], deep[i]);
}
maxr[n + 1] = 0;
for (int i = n; i >= 1; i--) {
maxr[i] = max(maxr[i + 1], deep[i]);
}
int m = (int)queries.size();
vector<int> ans(m);
for (int i = 0; i < m; i++) {
int leftMax = maxl[dfn[queries[i]] - 1];
int rightMax = maxr[dfn[queries[i]] + size0[dfn[queries[i]]]];
ans[i] = max(leftMax, rightMax);
}
return ans;
} int main() {
TreeNode node9(9);
TreeNode node8(8);
node8.right = &node9;
TreeNode node2(2);
TreeNode node4(4);
TreeNode node1(1);
TreeNode node6(6);
TreeNode node7(7);
node7.left = &node4;
node7.right = &node6;
TreeNode node3(3);
node3.left = &node1;
node3.right = &node7;
TreeNode node5(5);
node5.left = &node8;
node5.right = &node3;
vector<int> queries{ 3, 2, 4, 8 };
auto ans = treeQueries(&node5, queries);
cout << "The query results are: [";
for (int i = 0; i < ans.size(); i++) {
if (i > 0) {
cout << ", ";
}
cout << ans[i];
}
cout << "]" << endl;
return 0;
}

2023-05-03:给你一棵 二叉树 的根节点 root ,树中有 n 个节点 每个节点都可以被分配一个从 1 到 n 且互不相同的值 另给你一个长度为 m 的数组 queries 你必须在树上执行的更多相关文章

  1. 点击一个div ,把div里的某个参数的值,传到一个input里面

    ​​​

  2. C++ 推断一棵二叉树是否对称

    一棵二叉树对称,就是说它假设以根为轴,翻转过去一样.例如以下图所看到的,以虚线为轴.把左边翻转到右边,各顶点及顶点中的值一一相应. watermark/2/text/aHR0cDovL2Jsb2cuY ...

  3. 剑指offer17:输入两棵二叉树A,B,判断B是不是A的子结构。(ps:我们约定空树不是任意一个树的子结构)

    1 题目描述 输入两棵二叉树A,B,判断B是不是A的子结构.(ps:我们约定空树不是任意一个树的子结构) 2 思路和方法 (1)先在A中找和B的根节点相同的结点 (2)找到之后遍历对应位置的其他结点, ...

  4. 面试题:给定一个长度为N的数组,其中每个元素的取值范围都是1到N。判断数组中是否有重复的数字

    题目:给定一个长度为N的数组,其中每个元素的取值范围都是1到N.判断数组中是否有重复的数字.(原数组不必保留) 方法1.对数组进行排序(快速,堆),然后比较相邻的元素是否相同.时间复杂度为O(nlog ...

  5. CF E. Vasya and a Tree】 dfs+树状数组(给你一棵n个节点的树,每个点有一个权值,初始全为0,m次操作,每次三个数(v, d, x)表示只考虑以v为根的子树,将所有与v点距离小于等于d的点权值全部加上x,求所有操作完毕后,所有节点的值)

    题意: 给你一棵n个节点的树,每个点有一个权值,初始全为0,m次操作,每次三个数(v, d, x)表示只考虑以v为根的子树,将所有与v点距离小于等于d的点权值全部加上x,求所有操作完毕后,所有节点的值 ...

  6. 前端面试题:不使用loop循环,创建一个长度为100的数组,并且每个元素的值等于它的下标,,怎么实现好?

    昨天,看这道题,脑子锈住了,就是没有思路,没看明白是什么意思?⊙﹏⊙|∣今天早上起床,想到需要思考一下这个问题. 当然,我没想明白为什么要这样做?(创建一个长度为100的数组,并且每个元素的值等于它的 ...

  7. 2021.05.03 T3 数字

    2021.05.03 T3 数字 问题描述 一个数字被称为好数字当他满足下列条件: 1. 它有**2*n**个数位,n是正整数(允许有前导0) 2. 构成它的每个数字都在给定的数字集合S中. 3. 它 ...

  8. 算法进阶面试题05——树形dp解决步骤、返回最大搜索二叉子树的大小、二叉树最远两节点的距离、晚会最大活跃度、手撕缓存结构LRU

    接着第四课的内容,加入部分第五课的内容,主要介绍树形dp和LRU 第一题: 给定一棵二叉树的头节点head,请返回最大搜索二叉子树的大小 二叉树的套路 统一处理逻辑:假设以每个节点为头的这棵树,他的最 ...

  9. 剑指offer38:输入一棵二叉树,求该树的深度

    1 题目描述 输入一棵二叉树,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的长度为树的深度. 2 思路和方法 深度优先搜索,每次得到左右子树当前最大路径,选择 ...

  10. 如何一步一步用DDD设计一个电商网站(九)—— 小心陷入值对象持久化的坑

    阅读目录 前言 场景1的思考 场景2的思考 避坑方式 实践 结语 一.前言 在上一篇中(如何一步一步用DDD设计一个电商网站(八)—— 会员价的集成),有一行注释的代码: public interfa ...

随机推荐

  1. Linux 复制时排除某文件/目录

    如果要排除/home/data目录下面的a.b.c.三个目录,同时拷贝其它所有目录,执行rsync命令yum install rsync -y #安装rsync 排除单个文件/目录rsync -avP ...

  2. C#中Base64转换为byte[]再进行处理

    byte[] arr = Convert.FromBase64String(temp.Substring(temp.IndexOf(",") + 1)); using (Strea ...

  3. TCP 三次握手,给我长脸了噢

    大家好,我是小富~ 个人资源分享网站:FIRE 本文收录在 Springboot-Notebook 面试锦集 前言 之前有个小伙伴在技术交流群里咨询过一个问题,我当时还给提供了点排查思路,是个典型的八 ...

  4. 声网 VQA:将实时互动中未知的视频画质用户主观体验变可知

    在实时互动场景中,视频画质是影响观众体验的关键指标,但如何实时评价视频的画质一直是个行业难题,需要将未知的视频画质用户主观体验变成可知. 未知的部分往往是最需要攻克的,声网也一直在持续探索符合实时互动 ...

  5. University of Toronto Faculty of Arts and Science MAT344– Final Assessment Combinatorics Instructors: Stanislav Balchev and Max Klambauer 19 August 2020

    目录 随便找的一份测试题 T7 T9 T6 T5 solution to (a) solution to (b) solution to (c) solution to (d) T1 T2 T3 T4 ...

  6. 《操作系统导论》读书笔记1——CPU虚拟化,进程

    系列文章目录和关于我 一丶CPU的虚拟化 一个桃子,我们称之为物理(physical)桃子.但有很多想吃这个桃子的 人,我们希望向每个想吃的人提供一个属于他的桃子,这样才能皆大欢喜.我们把给每个 人的 ...

  7. vue 前端项目创建

    一.创建项目 将vue-admin-template-master 模板放入创建的 VS code 的工作空间.重命名为自己的项目. 模块获取方法:关注"Java程序员进阶",回复 ...

  8. SpringCloud Ribbon 负载均衡

    Spring Cloud Ribbon 是一个基于 HTTP 和 TCP 的客户端负载均衡工具.可以将面向服务的 REST 模板请求自动转化成客户端负载均衡的服务调用.Spring Cloud Rib ...

  9. mixins使用混入引入组件,并可以使用公共函数。组件类同名函数可以替代公共函数。使用$ref获得子元素数据和元素dom节点。使用$parents获得父元素数据。slot插槽的使用

    父组件: <template> <div class="box"> <Header > <div slot="left" ...

  10. python线程之event事件

    from threading import Thread, Event import time event = Event() def light(): print('红灯亮着,所有车都要等待') t ...