树与图的深度优先遍历——AcWing.846树的重心

树与图的深度优先遍历

定义

从图的某个顶点出发,沿着一条路径尽可能深地访问图中顶点。

运用情况

  • 图的连通性判断。
  • 寻找特定路径或回路。

注意事项

  • 要标记已访问的节点,以避免重复访问导致死循环。
  • 对于有向图和无向图可能需要不同的处理。

解题思路

  • 也是可以用递归方法,从起始顶点开始,访问该顶点,标记为已访问,然后对其未访问的邻接顶点进行深度优先遍历。

例如,对于一棵树,从根节点开始深度优先遍历,先访问根节点,再递归访问左子树和右子树。在图中,比如从一个顶点出发,访问相邻顶点,再递归深入访问它们的相邻顶点,直到无法继续深入为止,然后回溯。

AcWing.846树的重心

题目描述

846. 树的重心 - AcWing题库

运行代码

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
const int MAXN = 100005;
int n, root, ans, maxChild;
vector<int> G[MAXN];
int dfs1(int u, int fa) {int size = 1;for (int v : G[u]) {if (v != fa) {size += dfs1(v, u);}}return size;
}
int dfs2(int u, int fa, int tot) {int size = 1, maxSubtree = 0;for (int v : G[u]) {if (v != fa) {int subtreeSize = dfs2(v, u, tot);maxSubtree = max(maxSubtree, subtreeSize);size += subtreeSize;}}maxSubtree = max(maxSubtree, tot - size);ans = min(ans, maxSubtree);return size;
}
int main() {cin >> n;for (int i = 1; i < n; ++i) {int u, v;cin >> u >> v;G[u].push_back(v);G[v].push_back(u);}ans = n;dfs1(root = 1, 0);dfs2(root, 0, n);cout << ans << endl;return 0;
}

代码思路

找到一棵树的重心,并输出删除重心后,剩余各个连通块中点数的最大值。代码采用了两次深度优先搜索(DFS)的策略来实现这一目标。

  1. dfs1:第一次DFS是为了计算每个节点的子树大小(节点数)。从根节点开始,递归遍历每个子节点,并将子树的节点数累加到当前节点的子树大小中。

  2. dfs2:第二次DFS是为了确定删除每个节点后,剩余树的最大连通块大小。这次DFS不仅递归遍历每个子节点,还计算以当前节点为根的子树被删除后,剩余部分的最大连通块大小,并更新全局变量ans为所有可能情况中的最小值。

  3. 主函数:首先读取树的节点数n,然后建立无向图的邻接表表示。之后,从节点1开始进行两次DFS,最后输出ans作为删除重心后,剩余连通块中点数的最大值。

改进思路

  1. 初始化根节点:原代码中直接将root设为1,实际上在处理任意树结构时,可以选择任一节点作为起点,但最好明确指出这一点或者实现一个方法来自动选择一个合适的起点(如度最小的节点)。

  2. 减少全局变量使用:尽量减少全局变量的使用,可以将ansG作为参数传递给函数,提高代码的封装性和可重用性。

  3. 优化输出逻辑:在dfs2中直接记录并更新最小的最大连通块大小,避免不必要的全局变量更新。

改进代码(AI生成)

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
const int MAXN = 100005;
vector<int> G[MAXN];
pair<int, int> findCentroid(int u, int fa, int tot, vector<int>& subtreeSize) {int size = 1, maxSubtree = 0;for (int v : G[u]) {if (v != fa) {auto [subtreeSz, _] = findCentroid(v, u, tot, subtreeSize);size += subtreeSz;maxSubtree = max(maxSubtree, subtreeSz);}}maxSubtree = max(maxSubtree, tot - size);subtreeSize[u] = size;return {size, maxSubtree};
}
int main() {int n;cin >> n;for (int i = 1; i < n; ++i) {int u, v;cin >> u >> v;G[u].push_back(v);G[v].push_back(u);}vector<int> subtreeSize(n + 1, 0);auto [_, ans] = findCentroid(1, 0, n, subtreeSize);  cout << ans << endl;return 0;
}

其它代码

#include<iostream>
#include<cstring>
using namespace std;
const int N=100010;
int h[N],e[N*2],ne[N*2],idx,ans=N;
bool st[N];
int n;
void add(int a,int b)
{e[idx]=b;ne[idx]=h[a],h[a]=idx++;
}
int dfs(int u)
{st[u]=true;int size=0,sum=0;for(int i=h[u];i!=-1;i=ne[i]){int j=e[i];if(st[j]) continue;int s=dfs(j);size=max(size,s);sum+=s;}size=max(size,n-sum-1);ans=min(ans,size);return sum+1;
}
int main()
{memset(h, -1, sizeof h);cin>>n;for(int i=0;i<n-1;i++){int a,b;cin>>a>>b;add(a,b),add(b,a);}dfs(1);cout<<ans<<endl;return 0;
}

代码思路

数据结构与初始化

  1. 使用邻接表表示无向树结构。h[N]数组用于存储每个节点的边链表的头指针,初始值为-1;e[N*2]和ne[N*2]分别存储边的目标节点和下一条边的索引,idx用来追踪当前使用的边索引;st[N]数组标记节点是否已被访问过。
  2. N定义为节点数的上限,初始化为100010。
  3. 全局变量ans初始化为N,用于记录删除某个节点后,剩余各个连通块中点数的最大值,初始化为一个较大的值以便后续更新。

函数定义
add(int a, int b)函数用于添加边,构建无向图。参数a和b分别代表连接的两个节点。
主要逻辑

  1. 读取输入:读取节点数n,然后读取n-1条边的连接关系,构造无向树。
  2. 深度优先搜索(DFS):定义dfs(int u)函数,用于递归地遍历以节点u为根的子树,并计算以下内容:遍历子树中的每个节点,跳过已访问的节点。
  3. 计算每个子树的节点数(大小),并记录当前子树的最大节点数size。
  4. 计算以当前节点为根的子树的节点总和sum。
  5. 更新最大连通块的大小,考虑当前节点作为分割点时的情况,即size=max(size, n-sum-1),其中n-sum-1表示除了当前子树以外的其他节点数。
  6. 返回当前子树的节点数加1(包括根节点自己)。
  7. 寻找重心并输出结果:从根节点(这里假设为节点1)开始调用dfs函数。遍历结束后,ans中存储的就是删除任一节点后,剩余各个连通块中点数的最大值。最后输出ans。

优化与注意事项

  1. 代码中通过双向连边构建无向图,但在树结构中,实际上只需要单向连边即可,不过这不影响算法的正确性。
  2. 该算法通过一次深度优先遍历完成任务,时间复杂度为O(n),是较为高效的解法。
  3. 代码中没有显式地寻找并标识重心,而是直接计算并输出了删除任一节点后最不利情况下的连通块大小,这在逻辑上等效于找到重心的概念,但并未直接指出哪个节点是重心。

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

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

相关文章

C++ Primer Plus第五版笔记(p201-250)

第六章 函数&#xff08;下&#xff09; 在含有return语句的循环后面应该也有一条return语句 不要返回局部对象的引用或指针&#xff0c;当函数结束时临时对象占用的空间也就随之释放掉了&#xff0c;所以两条return语句都指向了不再可用的内存空间。 如果函数返回指针、引用…

排序(2)【选择排序】【快速排序】

一.选择排序 选择排序就是选择一个数组的最大的数字或者最小的数字&#xff0c;放在一整个数组的最后或者开头的位置。 1.选择排序的实现 我们可以对选择排序进行一些加强&#xff0c;普通的选择排序是选择最小的数&#xff0c;然后进行交换。这个加强之后就是我们既要选择出…

从ES的JVM配置起步思考JVM常见参数优化

目录 一、真实查看参数 &#xff08;一&#xff09;-XX:PrintCommandLineFlags &#xff08;二&#xff09;-XX:PrintFlagsFinal 二、堆空间的配置 &#xff08;一&#xff09;默认配置 &#xff08;二&#xff09;配置Elasticsearch堆内存时&#xff0c;将初始大小设置为…

ElasticSearch + kibana:类型声明

当我们使用 kibana 创建索引时&#xff0c;如果不申明数据类型&#xff0c;默认字符串赋予 text类型&#xff0c;如下图所示 接下来我们继续创建多条数据如下&#xff1a; 下面我们来检索下&#xff1a; 通过以上两个案例我们发现&#xff0c;使用 match 模糊查询 li-3 明明…

别再问别人了,这是小白都能懂的拓扑图指南

号主&#xff1a;老杨丨11年资深网络工程师&#xff0c;更多网工提升干货&#xff0c;请关注公众号&#xff1a;网络工程师俱乐部 上午好&#xff0c;我的网工朋友。 老杨的网工交流群里经常会有这种现象&#xff1a; 一群小伙伴在问各类型拓扑图的问题&#xff0c;怎么设计&…

monitor-zabbix

监控体系理论 学习本篇文章&#xff0c;了解运维监控系统的前世今生 zabbix官网仓库地址 zabbix官网 https://www.zabbix.com/cn/zabbix官网仓库地址 http://repo.zabbix.com/zabbix/ http://repo.zabbix.com/zabbix/4.0/ubuntu/pool/main/z/zabbix-release/zabbix-release_…

Hi3861 OpenHarmony嵌入式应用入门--基于HI-12F开发板烧写程序

首先需要一个开发板&#xff0c;我已经在嘉立创上进行了开源&#xff0c;基于安信可hi-12f模块的开发板&#xff0c;集成了两个按键&#xff0c;一个三色灯&#xff0c;一个滑动变阻器&#xff0c;可外接一个0.96寸液晶。 【HI-12F】基于安信可hi-12f模块的开发板 - 嘉立创EDA…

借助Aspose.Email,使用 C# .NET 创建 PST 文件并填充内容

PST&#xff08;个人存储表&#xff09;文件是管理 Outlook 数据的重要组成部分&#xff0c;方便存储电子邮件、联系人、日历和其他项目。在 C# .NET 开发领域&#xff0c;创建和管理存储文件的过程对于各种应用程序至关重要。 在本文中&#xff0c;我们将探讨如何使用 C# .NE…

内窥镜窄带光

文章目录 NBI相关信息 NBI相关信息 第一不知道哪家有这个技术&#xff1f; 第二直接搜索找不到相关信息 第三只能搜企业官网 搜集到的与NBI&#xff0c;相关的信息如下 英美达医疗公司 https://www.innermed.com/index.php/gongsixinwen/139.html 新光维医疗公司 官网页面…

【Spring】1. Maven项目管理

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 |《Web世界探险家》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更…

【无标题】Pycharm执行报错

file 读取未指定utf-8编码&#xff0c;加上就好了 疑问&#xff1a;为什么 有的电脑可以直接跑呢&#xff1f;该电脑、Pycharm、工程&#xff0c;已经做了修改设置默认值&#xff0c;但是到新的电脑上&#xff0c;就需要重新设置&#xff0c;所以 file 读、写&#xff0c;最好…

鸿蒙轻内核调测-内存调测-内存泄漏检测

1、基础概念 内存泄漏检测机制作为内核的可选功能&#xff0c;用于辅助定位动态内存泄漏问题。开启该功能&#xff0c;动态内存机制会自动记录申请内存时的函数调用关系&#xff08;下文简称LR&#xff09;。如果出现泄漏&#xff0c;就可以利用这些记录的信息&#xff0c;找到…

程序员日志之计算机相关专业还值得选择吗?

目录 传送门正文日志1、概要2、专业选择2.1、专业2.2、学校2.3、城市 3、计算机相关专业还值得选择吗&#xff1f; 传送门 SpringMVC的源码解析&#xff08;精品&#xff09; Spring6的源码解析&#xff08;精品&#xff09; SpringBoot3框架&#xff08;精品&#xff09; MyB…

中华老字号李良济,展现百年匠心之魅力,释放千年中医药文化自信

6月14-16日&#xff0c;“潮品老字号 国货LU锋芒”江苏老字号博览会在南京隆重启幕&#xff0c;中华老字号李良济凭借过硬的品牌实力和优质的口碑再次受邀参加&#xff0c;并在展会上绽放百年匠心魅力&#xff0c;彰显千年中医药文化自信&#xff01; 百年匠心 以实力铸就荣耀…

计算机组成原理之定点乘法运算

文章目录 原码并行乘法与补码并行乘法原码算法运算规则存在的问题带符号的阵列乘法器习题原码阵列乘法器间接补码阵列乘法器直接补码阵列乘法器 补码与真值的转换 原码并行乘法与补码并行乘法 原码算法运算规则 存在的问题 理解流水式阵列乘法器&#xff08;并行乘法器&#x…

Java环境安装

下载JDK https://www.oracle.com/cn/java/technologies/downloads/#jdk22-windows 点开那个下载都可以但是要记住下载的路径因为下一步要添加环境变量 选择编辑系统环境变量 点击环境变量 点击新建 新建环境变量JAVA_HOME 并输入JDK在计算机保存的路径 打开cmd 输入java -…

GStreamer——教程——基础教程4:Time management

基础教程4&#xff1a;Time management&#xff08;时间管理&#xff09; 目标 本教程展示了如何使用GStreamer时间相关工具。特别是&#xff1a; 如何查询管道以获取流位置或持续时间等信息。如何寻找&#xff08;跳转&#xff09;到流内的不同位置&#xff08;时间&#x…

数据结构与算法笔记:基础篇 -递归树:如何借助树来求解递归算法的时间复杂度?

概述 我们都知道&#xff0c;递归代码的时间复杂度分析起来很麻烦。在《排序(下)》哪里讲过&#xff0c;如何用递推公式&#xff0c;求解归并排序、快速排序的时间复杂度&#xff0c;但是有些情况&#xff0c;比如快排的平均时间复杂度的分析&#xff0c;用递推公式的话&#…

《天软股票特色因子定期报告》

最新《天软股票特色因子定期报告》&#xff08;2024-06&#xff09;&#xff0c;抢先发布 内容概要如下&#xff1a; 天软特色因子A08006&#xff08;近一月日度买卖压力2&#xff09;从行业角度分析&#xff0c;在电子设备、石油石化行业表现稳定&#xff0c;无论在有效性、区…

复星杏脉算法面经2024年5月16日面试

复星杏脉算法面经2024年5月 面试记录&#xff1a;3个部分1. 自己介绍 2. 问八股 3.代码题先自我介绍20分钟问问题1. 梯度爆炸怎么解决&#xff0c;三个解决方案&#xff1a;梯度裁剪&#xff08;Gradient Clipping&#xff09;正则化&#xff08;Regularization&#xff09;调整…