2024-04-21:用go语言,给一棵根为1的树,每次询问子树颜色种类数。 假设节点总数为n,颜色总数为m, 每个节点的颜色,依次给出,整棵树以1节点做头, 有k次查询,询问某个节点为头的子树,一共
2024-04-21:用go语言,给一棵根为1的树,每次询问子树颜色种类数。
假设节点总数为n,颜色总数为m,
每个节点的颜色,依次给出,整棵树以1节点做头,
有k次查询,询问某个节点为头的子树,一共有多少种颜色。
1 <= n, m, k <= 10^5。
答案2024-04-21:
来自左程云。
大体步骤如下:
大体过程描述:
1.数据结构初始化:定义全局变量和数组用来存储图的结构、节点颜色等信息,并初始化相关数组和变量。
2.输入处理:通过预定义的输入数组,按给定格式依次读取节点数n,建立树的连接关系,记录每个节点的颜色。
3.DFS遍历:
第一次DFS(dfs1):计算每个节点子树的大小,并标记每个节点的重节点。
第二次DFS(dfs2):处理每个节点的子树,包括处理重节点和非重节点的不同子树,更新颜色计数和子树的颜色种类数。
4.颜色计数:通过add函数和delete函数实现颜色的增加与减少操作,维护当前节点子树中颜色种类的计数。
5.输出查询结果:对于每次查询,按照给定节点进行处理,并输出计算得到的颜色种类数。
时间复杂度:
DFS1:对整个树进行一次DFS,时间复杂度为O(n)。
DFS2:同样对整个树进行一次DFS,时间复杂度为O(n)。
add和delete函数:每个节点至多被遍历4次(每条边两次),因此每次add和delete的时间复杂度为O(n)。
查询:对于每次查询,计算颜色种类数时需要遍历整个子树,时间复杂度为O(n)。
综上,总的时间复杂度为O(n)。
空间复杂度:
graph, color, size, heavy, cnt, ans:每个数组的长度为n,因此空间复杂度为O(n)。
其他局部变量:不超过常数大小,可忽略。
综上,总的额外空间复杂度为O(n)。
Go完整代码如下:
package main
import (
"fmt"
)
var MAXN int = 200005
var graph [][]int
var color []int
var size []int
var heavy []int
var cnt []int
var ans []int
var n, m int
func main() {
graph = make([][]int, MAXN)
for i := range graph {
graph[i] = make([]int, 0)
}
color = make([]int, MAXN)
size = make([]int, MAXN)
heavy = make([]int, MAXN)
cnt = make([]int, MAXN)
ans = make([]int, MAXN)
inputs := []int{5,
1, 2,
1, 3,
2, 4,
2, 5,
1, 2, 2, 3, 3,
5,
1,
2,
3,
4,
5}
ii := 0
for ii < len(inputs) {
n = inputs[ii]
ii++
for i := 1; i <= n; i++ {
graph[i] = make([]int, 0)
}
for i := 1; i < n; i++ {
a := inputs[ii]
ii++
b := inputs[ii]
ii++
graph[a] = append(graph[a], b)
graph[b] = append(graph[b], a)
}
for i := 1; i <= n; i++ {
c := inputs[ii]
ii++
color[i] = c
}
dfs1(1, 0)
dfs2(1, 0, false)
m = inputs[ii]
ii++
for i := 1; i <= m; i++ {
q := inputs[ii]
ii++
fmt.Println(ans[q])
}
}
}
var total int
func dfs1(cur, father int) {
size[cur] = 1
for _, next := range graph[cur] {
if next != father {
dfs1(next, cur)
size[cur] += size[next]
if size[heavy[cur]] < size[next] {
heavy[cur] = next
}
}
}
}
func dfs2(cur, father int, isHeavy bool) {
for _, next := range graph[cur] {
if next != father && next != heavy[cur] {
dfs2(next, cur, false)
}
}
if heavy[cur] != 0 {
dfs2(heavy[cur], cur, true)
}
add(cur, father, heavy[cur])
ans[cur] = total
if !isHeavy {
delete(cur, father)
total = 0
}
}
func add(cur, father, except int) {
cnt[color[cur]]++
if cnt[color[cur]] == 1 {
total++
}
for _, next := range graph[cur] {
if next != father && next != except {
add(next, cur, except)
}
}
}
func delete(cur, father int) {
cnt[color[cur]]--
for _, next := range graph[cur] {
if next != father {
delete(next, cur)
}
}
}
2024-04-21:用go语言,给一棵根为1的树,每次询问子树颜色种类数。 假设节点总数为n,颜色总数为m, 每个节点的颜色,依次给出,整棵树以1节点做头, 有k次查询,询问某个节点为头的子树,一共的更多相关文章
- 牛客练习赛47 E DongDong数颜色 (树状数组维护区间元素种类数)
链接:https://ac.nowcoder.com/acm/contest/904/E 来源:牛客网 DongDong数颜色 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 5242 ...
- 21世纪C语言(影印版)
<21世纪C语言(影印版)> 基本信息 原书名:21st Century C 作者: Ben Klemens 出版社:东南大学出版社 ISBN:9787564142056 上架时间:201 ...
- 用css属性画出一棵圣诞树
对于学习前端的童鞋,css的掌握是必须的.今天就来实现用css画出一棵圣诞树. 主要练习的是css里面border的练习与掌握程度. 在body创建一个主区域<div></div&g ...
- Ubuntu 12.04上安装R语言
Ubuntu 12.04上安装R语言 作者:凯鲁嘎吉 - 博客园 http://www.cnblogs.com/kailugaji/ R的安装 sudo gedit /etc/apt/sources. ...
- 剑指offer19:按照从外向里以顺时针的顺序依次打印出每一个数字,4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2,3,4,8,12,16,15,14,13,9,5,6,7,11,10.
1 题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印 ...
- Ubuntu 18.04 安装配置 go 语言
Ubuntu 18.04 安装配置 go 语言 1.下载 下载 jdk 到 Downloands 文件夹下 cd 进入 /usr/local, 创建 go 文件夹, 然后 cd 进这个文件夹 cd / ...
- [BZOJ3110] [Zjoi2013] K大数查询 (树套树)
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置 ...
- BZOJ3110 K大数查询 【线段树 + 整体二分 或 树套树(非正解)】
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位 ...
- 树套树专题——bzoj 3110: [Zjoi2013] K大数查询 & 3236 [Ahoi2013] 作业 题解
[原题1] 3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 978 Solved: 476 Descri ...
- BZOJ 3110 [Zjoi2013]K大数查询 (CDQ分治+树状数组)
题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是 ...
随机推荐
- 软件发布时 生成发布日志文件 单点登录 getGitInfo.bat
需求 每次发包的时候,前端是3个包,如果后期出现问题,不好回查 所以把当前项目的git信息记录下来 以便回查 第一次手动写了下,发现比较麻烦,所以写个脚本,每次发布的时候 运行下即可 上代码 软件发布 ...
- etcd每个节点都存储了完整的键值对数据集,为什么扩容etcd集群仍可分散存储压力?
etcd每个节点都存储了完整的键值对数据集,这主要是为了确保数据的一致性和高可用性.在这种设计下,任何一个节点都可以处理读取请求,并在本地提供数据,从而无需跨节点通信.这种冗余的数据存储方式也增加了系 ...
- ubuntu添加桌面快捷打开方式
不太喜欢ubuntu开机后空荡荡的桌面,希望可以有些像windows一样的快捷打开方式.看了一些博客,也自己探索了一下,发现了在ubuntu中添加软件自带的桌面快捷打开方式的方法. 在终端 cd /u ...
- maven解决尝试手段
发现原来用的buildBody不好用,百度这个请求有异于其他sdk 关于写身份证接口那边:首先报有两个slf4j冲突,经过查阅,不能包含两个slf4j遂写了exclusion,但是排除不了,要使用** ...
- netty Recycler对象池
前言 池化思想在实际开发中有很多应用,指的是针对一些创建成本高,创建频繁的对象,用完不弃,将其缓存在对象池子里,下次使用时优先从池子里获取,如果获取到则可以直接使用,以此降低创建对象的开销. 我们最熟 ...
- 对TCP/IP协议的理解
话说两台电脑要通讯就必须遵守共同的规则,就好比两个人要沟通就必须使用共同的语言一样.一个只懂英语的人,和一个只懂中文的人由于没有共同的语言(规则)就没办法沟通.两台电脑之间进行通讯所共同遵守的规则,就 ...
- 基于R语言的GD库实现地理探测器并自动将连续变量转为类别变量
本文介绍基于R语言中的GD包,依据栅格影像数据,实现自变量最优离散化方法选取与执行,并进行地理探测器(Geodetector)操作的方法. 首先,在R语言中进行地理探测器操作,可通过geode ...
- vue-router动态注册
来源 写路由时每新建一个路由都需要import一下或其他方式(如箭头函数import)很是麻烦,有麻烦就有需求,于是以下这篇文章就来了 吹水 要想动态注册路由,那么就需要制定规则,即每个路由有一定的规 ...
- vue三种插槽
1. 作用:让父组件可以向子组件指定位置插入html结构,也是一种组件间通信的方式,适用于 父组件 ===> 子组件 . 2. 分类:默认插槽.具名插槽.作用域插槽 3. 使用方式: a.默认插 ...
- NF-ResNet:去掉BN归一化,值得细读的网络信号分析 | ICLR 2021
论文提出NF-ResNet,根据网络的实际信号传递进行分析,模拟BatchNorm在均值和方差传递上的表现,进而代替BatchNorm.论文实验和分析十分足,出来的效果也很不错.一些初始化方法的理论效 ...