并查集

Posted 牧马人夏峥

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了并查集相关的知识,希望对你有一定的参考价值。

刷题是遇到一题:http://www.nowcoder.com/questionTerminal/b1303e827e7f4df4a816598d008bbe72

 

这道题要用到并查集来解(Union-Find),这位网友总结的很详细:http://blog.csdn.net/dm_vincent/article/details/7655764

并查集主要包括两部分,find和union函数。其中find函数找根结点,union函数合并结点。根据初始化值不同,find和union函数的写法

也略有不同。

下面是某网友写的答案,略有瑕疵,我修改了下,加上点注释:

int find(vector<int>& id, int i)
{
    //只有根结点或者未连接的点为负,其他点为其根结点的索引值
    //如果不是根结点,找到根结点
    while (id[i] >= 0)i = id[i];

    return i;
}

void unioned(vector<int>&id, int i, int j)
{

    //将根值更小的点设为根结点
    if (id[i] < id[j])
    {
        id[i] += id[j];
        id[j] = i;//直接保存根结点的索引
    }
    else
    {
        id[j] += id[i];
        id[i] = j;
    }
}
int main()
{
    string str;
    int count;//输入连通集个数
    while (cin >> str)
    {
        int len = str.length();
        //全部初始化为-1.
        vector<int> id(len, -1);
        cin >> count;
        for (int i = 0; i < count; i++)
        {
            int p, q;
            cin >> p >> q;
            int pRoot = find(id, p);
            int qRoot = find(id, q);
            if (pRoot != qRoot)
                unioned(id, pRoot, qRoot);//注意:是根结点相连接
        }
        vector<vector<int>> mat(len);
        for (int i = 0; i < id.size(); i++)
        {
            if (id[i] == -1)continue;

            int root = find(id, i);
            mat[root].push_back(i);//保存各个连通集的位置索引(按根结点)
        }
        //
        for (int i = 0; i < mat.size(); i++)
        {
            vector<char> chs;//保存连通集上的字符
            //如果不是根结点,则直接跳过
            for (int j = 0; j < mat[i].size(); j++)
            {
                chs.push_back(str[mat[i][j]]);
            }
            sort(chs.begin(), chs.end());//将同个连通集上的字符排序
            //写回字符串中
            for (int j = 0; j < mat[i].size(); j++)
                str[mat[i][j]] = chs[j];
        }
        cout << str;
    }
}
View Code

 

以上是关于并查集的主要内容,如果未能解决你的问题,请参考以下文章

想要学会并查集吗?看我四十行代码实现它

树--12---并查集

笔记并查集---无向图处理代码模板及类型题

并查集

力扣 每日一题 886. 可能的二分法难度:中等,rating: 1794(并查集 / 拆点优化的扩展域并查集)

并查集