AcWing 837. 连通块中点的数量

题目描述


分析:

本题是对并查集的变形应用,如果对并查集的基本概念了解不够透彻的可以看复习一下:并查集的递推和递归写法。先看 Q1 操作,就是并查集那查找的基本操作,不再赘述。本题的变形之处在于给集合附加了“大小”属性,即在维护 father 数组的同时也要维护一个 ssize 数组,其中 ssize[根结点] 存储了该根结点下集合内元素的数量,而 ssize 的初始化也比较好理解,即我们之前提到的:初始时每个元素都是独立的一个集合,所以ssize[i] = 1 ( 1 ≤ i ≤ N ) (1 ≤ i ≤ N) (1iN)。对于 Q2 操作,我想思路已经很清楚了,直接查询给定的根结点信息并输出 ssize[father[input]] 即可。但还有另外一点就是 C 操作(合并两个集合)时,合并后的连通块大小该如何更新呢?假设我们需要合并两连通块,那么最重要的就是要找到两连通块的根结点,设为 findFather(a) findFather(a)。然后将 findFather(a) 连通块大小加到 findFather(a) 的连通块上即可。


代码(C++)

#include <iostream>

using namespace std;

const int N = 100010;
int father[N], ssize[N];

int findFather(int x)
{
    if (x == father[x]) return x; // 直到找到根结点
    else
    {
        int F = findFather(father[x]); // 等式右边找到根结点后开始回溯
        father[x] = F; // 路径上的每个结点的父结点都设置为根结点
        return F;
    }
}

int main()
{
    int n, m;
    cin >> n >> m;
    
    for (int i = 1; i <= n; i ++)
    {
        father[i] = i;
        ssize[i] = 1; // 初始化,使所有连通块的大小都为 1
    }
    
    while (m --)
    {
        string op;
        int a, b;
        cin >> op;
        
        if (op == "C")
        {
            cin >> a >> b;
            if (findFather(a) != findFather(b))
            {
                /*这里是易错点,如果没有题前将a b集合的根结点取出来的操作
                    例如 a = findFather(a) b = findFather(b),
                    那就一定要按照先更改连通块大小再进行集合合并的操作!!
                    若没有取出根结点又颠倒操作,则会导致更改连通块大小时根结点的重叠
                    导致答案出错
                */
                // 接入 b 的集合后,b 应该加上 a 集合的连通块的大小
                ssize[findFather(b)] += ssize[findFather(a)];
                // 将 a 为根结点的集合接到 b 为根结点的集合下
                father[findFather(a)] = findFather(b); 
            }
        }
        else if (op == "Q1")
        {
            cin >> a >> b;
            if (findFather(a) == findFather(b)) puts("Yes");
            else puts("No");
        }
        else
        {
            cin >> a;
            cout << ssize[findFather(a)] << endl;
        }
        
    }
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值