树上启发式合并(dsu on tree)学习

2024-04-13 23:36

本文主要是介绍树上启发式合并(dsu on tree)学习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

声明:本文部分内容摘自OI Wiki网站。详情可自行查看学习。

洛谷 P9233

  题目实际上是蓝桥杯 2023 年 A 组省赛的一道题。题干大致的意思是,给定一个含有 n n n 个结点,并且以 1 1 1 为根的一棵树,每个节点 i i i 都有一个颜色 C i C_i Ci,问这棵树有多少个子树是颜色平衡树。其中,如果一棵树中存在的每种颜色的结点个数都相同,则我们称它是一棵颜色平衡树。
  输入:一行一个 n n n,后面的 n n n 行每行一个数对 ( C i , F i ) (C_i,F_i) (Ci,Fi) 分别表示结点 i i i 的颜色和父亲结点。(保证 F 1 = 0 F_1=0 F1=0)。
  输出:这棵树有多少棵子树是颜色平衡树。

题目分析

  感觉题目最大的难点是, C i ≤ 200000 C_i\leq 200000 Ci200000。因为我们最朴素的思想就是:看一棵树是否是颜色平衡树,需要将其子树的颜色情况统计起来。比如,开一个数组c[1..N],其中子树中颜色为i的结点有c[i]个。通过统计一棵树所有子树的c可以得到这棵树的c
  但是现在有一个问题,如果按照上面的思路,显然有 N ≥ C i N\geq C_i NCi;如果每碰到一个子树就开一个c[N],肯定会MLE,并且由于每次合并子树的颜色情况时需要遍历这个数组c,多半也会TLE。
  于是我们又产生了一个想法,用一个map来代替c[N]。因为不是每棵子树都会出现所有的颜色,我们开map不会为没有出现的颜色分配空间,显然空间使用就大大减少了。这种情况下合并子树的颜色情况时,只需要遍历所有子树的map,一一加到树上就行了。
  经过实践,使用map的方法确实不会MLE,但是它TLE了(70 pts)。

  接触到了树上启发式合并这一思想之后,迅速写代码,于是AC。树上启发式合并的核心思想就是保留重儿子。在下面的代码中有注释说明。

AC代码

#include<iostream>using namespace std;int n,c[200005],ccnt[200005],cccnt,sz[200005],ncolor[200005],ecnt=1,fedge[200005],ledge[200005],ans,bigchild[200005];
struct {int end,next;
}edge[200005];void buildarc(int begin,int end){if(!begin)return;if(!fedge[begin])fedge[begin]=ledge[begin]=ecnt;else{edge[ledge[begin]].next=ecnt;ledge[begin]=ecnt;}edge[ecnt++].end=end;
}inline void addcc(int cc){if(!ccnt[cc])cccnt++;ccnt[cc]++;
}
inline void delcc(int cc){ccnt[cc]--;if(!ccnt[cc])cccnt--;
}void add(int node){if(c[ncolor[node]])delcc(c[ncolor[node]]);addcc(c[ncolor[node]]+1);c[ncolor[node]]++;
}
void del(int node){c[ncolor[node]]--;delcc(c[ncolor[node]]+1);if(c[ncolor[node]])addcc(c[ncolor[node]]);
}inline int isbalanced(){return cccnt==1?1:0;}// dfs0 用来求每个子树的大小的,也就是这个子树有多少个结点。
int dfs0(int cur,int father){int childsize,maxchild=0;sz[cur]=1;for(int e=fedge[cur];e;e=edge[e].next){if(edge[e].end==father)continue;sz[cur]+=(childsize=dfs0(edge[e].end,cur));if(childsize>maxchild){maxchild=childsize;bigchild[cur]=edge[e].end;}}return sz[cur];
}// 这个是用来加入/删除某一棵树的,mod == 1 是加入,mod == 0 是删除。
void dfs2(int cur,int father,bool mod){//1 add 0 delif(mod)add(cur);elsedel(cur);for(int e=fedge[cur];e;e=edge[e].next)if(edge[e].end!=father)dfs2(edge[e].end,cur,mod);
}
/***dfs1 是树上启发式合并的核心算法。一个数有多个子树,其中最大的子树的树根成为这个树的重儿子,其它的儿子是轻儿子。树上启发式算法在每个子树上的操作分为 3 步:1.遍历所有轻儿子子树,查看情况,不保留轻儿子子树对原树 c[N] 数组的影响。2.查看重儿子子树的情况,并且保留该子树对原树 c[N] 数组的影响。3.重新加入所有轻儿子子树,从而得到原树的情况。@param cur 当前结点
@param father 当前结点的父节点,用 father = 0 表示没有父节点
@param remain 是否要保留 cur 为根的子树对其父树的影响。也即 cur 是否是重儿子。
***/
void dfs1(int cur,int father,bool remain){//遍历所有轻儿子for(int e=fedge[cur];e;e=edge[e].next)if(edge[e].end!=father && edge[e].end!= bigchild[cur])dfs1(edge[e].end,cur,false);//查看重儿子if(bigchild[cur])dfs1(bigchild[cur],cur,true);//加入根节点add(cur);//重新加入所有轻儿子子树for(int e=fedge[cur];e;e=edge[e].next)if(edge[e].end!=father && edge[e].end!= bigchild[cur])dfs2(edge[e].end,cur,true);ans+=isbalanced();//如果要求不保留影响,cur 为根的树的所有贡献if(!remain)dfs2(cur,father,false);
}int main(){int f;cin>>n;for(int i=1;i<=n;i++){scanf("%d%d",&ncolor[i],&f);buildarc(f,i);}return dfs0(1,0),dfs1(1,0,true),cout<<ans,0;
} 

  AC代码中保留了c[N]数组,其含义与上文所述相同,即颜色的出现次数。可以看到代码中还出现了ccnt数组和cccnt变量。这是一个比较精妙的处理方式,可以在 O ( 1 ) O(1) O(1) 的时间复杂度内判断一棵树是否是颜色平衡树。ccnt[i] = j表示当前共有j个颜色的c值是i,即颜色出现次数的出现次数cccnt颜色出现次数的出现次数的出现次数,当且仅当cccnt == 1的时候,该树是一颗颜色平衡树。
  上面的描述可能有些烧脑,可以用题目给的样例来说明。样例输入:

6
2 0
2 1
1 2
3 3
3 4
1 4

  读者可以根据输入的含义自行画图。对于根节点1而言,颜色1,2,3分别出现了2,2,2次,所以c[1,2,3] = {2,2,2}。根据定义,有3个颜色的出现次数是2,所以ccnt[2] = 3。而对于其它的i != 2,都有cccnt[i] = 0,所以cccnt = 1,因此该树是颜色平衡树。

这篇关于树上启发式合并(dsu on tree)学习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/901529

相关文章

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

使用Python合并 Excel单元格指定行列或单元格范围

《使用Python合并Excel单元格指定行列或单元格范围》合并Excel单元格是Excel数据处理和表格设计中的一项常用操作,本文将介绍如何通过Python合并Excel中的指定行列或单... 目录python Excel库安装Python合并Excel 中的指定行Python合并Excel 中的指定列P

基于C#实现PDF文件合并工具

《基于C#实现PDF文件合并工具》这篇文章主要为大家详细介绍了如何基于C#实现一个简单的PDF文件合并工具,文中的示例代码简洁易懂,有需要的小伙伴可以跟随小编一起学习一下... 界面主要用于发票PDF文件的合并。经常出差要报销的很有用。代码using System;using System.Col

Python视频剪辑合并操作的实现示例

《Python视频剪辑合并操作的实现示例》很多人在创作视频时都需要进行剪辑,本文主要介绍了Python视频剪辑合并操作的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录介绍安装FFmpegWindowsMACOS安装MoviePy剪切视频合并视频转换视频结论介绍

不删数据还能合并磁盘? 让电脑C盘D盘合并并保留数据的技巧

《不删数据还能合并磁盘?让电脑C盘D盘合并并保留数据的技巧》在Windows操作系统中,合并C盘和D盘是一个相对复杂的任务,尤其是当你不希望删除其中的数据时,幸运的是,有几种方法可以实现这一目标且在... 在电脑生产时,制造商常为C盘分配较小的磁盘空间,以确保软件在运行过程中不会出现磁盘空间不足的问题。但在

在C#中合并和解析相对路径方式

《在C#中合并和解析相对路径方式》Path类提供了几个用于操作文件路径的静态方法,其中包括Combine方法和GetFullPath方法,Combine方法将两个路径合并在一起,但不会解析包含相对元素... 目录C#合并和解析相对路径System.IO.Path类幸运的是总结C#合并和解析相对路径对于 C

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;