zoukankan      html  css  js  c++  java
  • 并查集

    刷题是遇到一题: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
  • 相关阅读:
    src和href属性的区别
    cookie, sessionStorage和localStorage的区别
    C# base64 转 byte[]
    C# 中字符串string和字节数组byte[]的转换
    C#获取当前路径的七种方法 【转载】
    C# 读取大文件 (可以读取3GB大小的txt文件)
    伪装文件到图片工具
    nmap命令扫描存活主机
    .deb文件如何安装,Ubuntu下deb安装方法图文详解
    kali修改更新源及更新
  • 原文地址:https://www.cnblogs.com/573177885qq/p/5845576.html
Copyright © 2011-2022 走看看