博客
关于我
算法笔记----好朋友(并查集问题)
阅读量:567 次
发布时间:2019-03-09

本文共 2054 字,大约阅读时间需要 6 分钟。

并查集(Union-Find)是一种高效的数据结构,广泛应用于处理动态的连通性问题。以下是优化过的技术文档,基于用户的要求已进行适当修改,确保内容具有专业性和可读性。

并查集优化解法

问题分析

我们需要处理多组朋友关系,并找到每组的根节点数目。同一个集合的所有节点应共享一个根节点,最终统计这些根节点的数量即可。

核心思想

  • 并查集:维护每个节点的父节点,以及一个标记数组记录是否为根节点。
  • 路径压缩:在查找根节点时,压缩路径,使后续查找更高效。
  • 合并操作:将不同集合中的节点合并到一起。
  • 优化方案

    路径压缩

    在查找根节点时,将当前节点的父节点指向其祖父。这样做可以减少查找时间,因为每次查找时路径会被缩短。

    工作流程

  • 初始化:每个节点的父节点指向自己,标记其是否为根节点。
  • 处理输入:逐个处理朋友关系,执行合并操作。
  • 统计集合数:遍历所有节点,统计根节点数量。
  • 技术细节

    findFather函数

    int findFather(int x) {    if (father[x] != x) {        father[x] = findFather(father[x]);    }    return father[x];}
    • 递归查找:递归地查找节点的祖父,直到找到根节点。
    • 路径压缩:将当前节点的父节点指向根节点,减少后续查找的路径长度。

    Union函数

    void Union(int a, int b) {    int fa = findFather(a);    int fb = findFather(b);        if (fa != fb) {        father[fa] = fb;    }}
    • 合并操作:找到两节点的根节点,如果不同,则将一棵树合并到另一棵树的根节点下。
    • 无需额外结构:仅需修改父节点即可高效地管理集合。

    初始化函数

    void init(int n) {    for (int i = 1; i <= n; ++i) {        father[i] = i;    }}
    • 初始化父节点:每个节点的父节点初始化为自身,标记为非根节点。

    样例分析

    • 输入样例:7 51 22 33 11 45 6
    • 处理过程:逐个处理每对朋友关系,合并相应的集合。
    • 结果:最终有3个独立的集合。

    代码实现

    #include 
    #include
    #include
    #include
    #include
    using namespace std;const int maxn = 101;int father[maxn];int isRoot[maxn]; // 是否为根节点标记int findFather(int x) { if (father[x] != x) { father[x] = findFather(father[x]); } return father[x];}void Union(int a, int b) { int fa = findFather(a); int fb = findFather(b); if (fa != fb) { father[fa] = fb; }}void init(int n) { for (int i = 1; i <= n; ++i) { father[i] = i; isRoot[i] = false; }}int main() { cin >> n >> m; init(n); for (int i = 0; i < m; ++i) { int a, b; cin >> a >> b; Union(a, b); } for (int i = 1; i <= n; ++i) { isRoot[findFather(i)] = true; } int ans = 0; for (int i = 1; i <= n; ++i) { if (isRoot[i]) { ans++; } } cout << ans << endl; return 0;}

    优化效果

  • 路径压缩:通过将路径直接连接到根节点,减少查找时间。
  • 总体复杂度:时间复杂度接近O(α(n)),α为阿克曼函数,增长极慢。
  • 空间复杂度:O(n),适合处理大规模数据。
  • 通过这段优化代码,可以高效地解决类似问题,找到每个集合的根节点数目。该方案适用于动态数据的连通性问题,是数据结构和算法学习中的基础内容。

    转载地址:http://ndapz.baihongyu.com/

    你可能感兴趣的文章
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_插入修改删除增量数据实时同步_通过分页解决变更记录过大问题_01----大数据之Nifi工作笔记0053
    查看>>
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表或全表增量同步_实现指定整库同步_或指定数据表同步配置_04---大数据之Nifi工作笔记0056
    查看>>
    NIFI1.23.2_最新版_性能优化通用_技巧积累_使用NIFI表达式过滤表_随时更新---大数据之Nifi工作笔记0063
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_根据binlog实现数据实时delete同步_实际操作04---大数据之Nifi工作笔记0043
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置binlog_使用处理器抓取binlog数据_实际操作01---大数据之Nifi工作笔记0040
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置数据路由_实现数据插入数据到目标数据库_实际操作03---大数据之Nifi工作笔记0042
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置数据路由_生成插入Sql语句_实际操作02---大数据之Nifi工作笔记0041
    查看>>
    NIFI从MySql中离线读取数据再导入到MySql中_03_来吧用NIFI实现_数据分页获取功能---大数据之Nifi工作笔记0038
    查看>>
    NIFI从MySql中离线读取数据再导入到MySql中_不带分页处理_01_QueryDatabaseTable获取数据_原0036---大数据之Nifi工作笔记0064
    查看>>
    NIFI从MySql中离线读取数据再导入到MySql中_无分页功能_02_转换数据_分割数据_提取JSON数据_替换拼接SQL_添加分页---大数据之Nifi工作笔记0037
    查看>>
    NIFI从PostGresql中离线读取数据再导入到MySql中_带有数据分页获取功能_不带分页不能用_NIFI资料太少了---大数据之Nifi工作笔记0039
    查看>>
    nifi使用过程-常见问题-以及入门总结---大数据之Nifi工作笔记0012
    查看>>
    NIFI分页获取Mysql数据_导入到Hbase中_并可通过phoenix客户端查询_含金量很高的一篇_搞了好久_实际操作05---大数据之Nifi工作笔记0045
    查看>>
    NIFI分页获取Postgresql数据到Hbase中_实际操作---大数据之Nifi工作笔记0049
    查看>>
    NIFI同步MySql数据_到SqlServer_错误_驱动程序无法通过使用安全套接字层(SSL)加密与SQL Server_Navicat连接SqlServer---大数据之Nifi工作笔记0047
    查看>>
    Nifi同步过程中报错create_time字段找不到_实际目标表和源表中没有这个字段---大数据之Nifi工作笔记0066
    查看>>
    NIFI大数据进阶_FlowFile拓扑_对FlowFile内容和属性的修改删除添加_介绍和描述_以及实际操作---大数据之Nifi工作笔记0023
    查看>>
    NIFI大数据进阶_FlowFile生成器_GenerateFlowFile处理器_ReplaceText处理器_处理器介绍_处理过程说明---大数据之Nifi工作笔记0019
    查看>>
    NIFI大数据进阶_Json内容转换为Hive支持的文本格式_操作方法说明_01_EvaluteJsonPath处理器---大数据之Nifi工作笔记0031
    查看>>
    NIFI大数据进阶_Kafka使用相关说明_实际操作Kafka消费者处理器_来消费kafka数据---大数据之Nifi工作笔记0037
    查看>>