数据结构——并查集

打印 上一主题 下一主题

主题 962|帖子 962|积分 2886

目录
一并查集原理
二并查集实现
 三并查集应用
省份数量
等式方程的可满意性

一并查集原理

在一些应用问题中,必要将n个不同的元素划分成不相交的集合;                                                 开始时,每个元素自成一个单元素集合,然后按肯定的规律将不同集合举行合并;                        在此过程中要查询某一个元素归属于那个集合的运算。这类问题可以用并查集来办理
好比:某公司今年校招全国总共招生5人,西安招2人,成都招3人;5个人来自不同的学校,起先互不相识,每个学生都是一个独立的小团体;                                                       
现给这些学生举行编号:{0, 1, 2, 3,4}; (-1代表在该集合中只有自己一人)

毕业后,学生们要去公司上班,每个地方的学生自发组织成小分队一起上路,于是:
西安学生小分队s1={0,1},成都学生小分队s2={2,3,4},各自形成一个小团体:

   每棵树的根/每个集合的第一个数存的是集合的个数(为了与编号区别开存的是相反数),而在集合内的别的数则存的是第一个数的编号(通常是下标)
  通过平常的相处,2号与1号渐渐熟悉成为了朋友,此时这两个集合就可以合并成一起,成为一个新的集合:(哪个集合合并哪个集合随意,没规定)
 
   总结:
  1. 数组的下标对应集合中元素的编号;
2. 数组的值中如果为负数,负号代表根,数字代表该集合中元素个数;
3. 数组的值中如果为非负数,代表该元素双亲在数组中的下标(编号)
  二并查集实现

创建一个并查集对象通常必要根据传过来的数据个数举行开发空间并初始化成-1;实现时可以用vector来接收;(如果传过来是别的数据,我们就要在内部对数据举行编号)
   并查集实现一般实现四个功能接口:
  1找根;2合并两个集合;3判断是否在同一个集合;4.当前集合的个数
  1. #include <iostream>
  2. #include <vector>
  3. #include <unordered_map>
  4. using namespace std;
  5. // 如果是其它数据的话要自己建立映射
  6. //  template <class T>
  7. //  class UnionFindSet
  8. //  {
  9. //  public:
  10. //      UnionFindSet(const vector<T> &v)
  11. //      {
  12. //          for (int i = 0; i < v.size(); i++)
  13. //          {
  14. //              _v.push_back(v[i]);
  15. //              _ufs[v[i]] = i;
  16. //          }
  17. //      }
  18. // private:
  19. //     vector<T> _v;               // 值映射下标
  20. //     unordered_map<T, int> _ufs; // 下标找值
  21. // };
  22. template <class T>
  23. class UnionFindSet
  24. {
  25. public:
  26.     UnionFindSet(int size)
  27.     {
  28.         _ufs.resize(size, -1);
  29.     }
  30.     // 找根
  31.     int FindSet(int x)
  32.     {
  33.         int root = x;
  34.         while (_ufs[root] >= 0)
  35.         {
  36.             root = _ufs[root]; // 更新根,一定注意不要写错
  37.         }
  38.         return root;
  39.     }
  40.     // 合并集合
  41.     void UnionSet(int x, int y)
  42.     {
  43.         int root1 = FindSet(x);
  44.         int root2 = FindSet(y);
  45.         if (root1 != root2)
  46.         {
  47.             _ufs[root1] += _ufs[root2];
  48.             _ufs[root2] = root1;
  49.         }
  50.     }
  51.     // 是否在同一集合
  52.     bool IsSet(int x, int y)
  53.     {
  54.         return FindSet(x) == FindSet(y);
  55.     }
  56.     // 当前集合个数
  57.     int Size()
  58.     {
  59.         int n = 0;
  60.         for (auto e : _ufs)
  61.         {
  62.             if (e < 0) // n-=e;
  63.                 ++n;
  64.         }
  65.         return n;
  66.     }
  67. private:
  68.     vector<T> _ufs;
  69. };
复制代码
 三并查集应用

省份数量

有了上面实现,办理该题不是轻而易举吗!
  1. #include <iostream>
  2. #include <vector>
  3. using namespace std;
  4. template <class T>
  5. class UnionFindSet
  6. {
  7. public:
  8.     UnionFindSet(int size)
  9.     {
  10.         _ufs.resize(size, -1);
  11.     }
  12.     int FindSet(int x)
  13.     {
  14.         int root = x;
  15.         while (_ufs[root] >= 0)
  16.         {
  17.             root = _ufs[root];
  18.         }
  19.         return root;
  20.     }
  21.     void UnionSet(int x, int y)
  22.     {
  23.         int root1 = FindSet(x);
  24.         int root2 = FindSet(y);
  25.         if (root1 != root2)
  26.         {
  27.             _ufs[root1] += _ufs[root2];
  28.             _ufs[root2] = root1;
  29.         }
  30.     }
  31.     bool IsSet(int x, int y)
  32.     {
  33.         return FindSet(x) == FindSet(y);
  34.     }
  35.     int Size()
  36.     {
  37.         int n = 0;
  38.         for (auto e : _ufs)
  39.         {
  40.             if (e < 0) // n-=e;
  41.                 ++n;
  42.         }
  43.         return n;
  44.     }
  45. private:
  46.     vector<T> _ufs;
  47. };
  48. class Solution
  49. {
  50. public:
  51.     int findCircleNum(vector<vector<int>> &isConnected)
  52.     {
  53.         UnionFindSet<int> ufs(isConnected.size());
  54.         for (int i = 0; i < isConnected.size(); i++)
  55.         {
  56.             for (int j = 0; j < isConnected[i].size(); j++)
  57.             {
  58.                 if (isConnected[i][j] == 1)
  59.                 {
  60.                     ufs.UnionSet(i, j);
  61.                 }
  62.             }
  63.         }
  64.         return ufs.Size();
  65.     }
  66. };
复制代码
等式方程的可满意性

   先遍历一遍:相等的字符存在同一个集合中;
  再遍历一遍:两个不相等的字符去并查集找是否在同一个集合中,是返回false;
  留意:办理该问题并不愿定每次都要把并查集实现一遍:实现出我们想要的函数即可
  1. class Solution
  2. {
  3. public:
  4.     bool equationsPossible(vector<string> &equations)
  5.     {
  6.         vector<int> ufs(26, -1);
  7.         auto Find = [&ufs](int x)
  8.         {
  9.             while (ufs[x] >= 0)
  10.                 x = ufs[x]; // 注意这里的逻辑!
  11.             return x;
  12.         };
  13.         for (auto &equation : equations)
  14.         {
  15.             if (equation[1] == '=')
  16.             {
  17.                 int root1 = Find(equation[0] - 'a'), root2 = Find(equation[3] - 'a');
  18.                 if (root1 != root2)
  19.                 {
  20.                     ufs[root1] += ufs[root2];
  21.                     ufs[root2] = root1; // 这里也要赋值对
  22.                 }
  23.             }
  24.         }
  25.         for (auto &equation : equations)
  26.         {
  27.             if (equation[1] == '!')
  28.             {
  29.                 int x = Find(equation[0] - 'a');
  30.                 int y = Find(equation[3] - 'a');
  31.                 if (x == y)
  32.                     return false;
  33.             }
  34.         }
  35.         return true;
  36.     }
  37. };
复制代码
   以上便是全部内容,有问题欢迎在评论区指正,感谢观看!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

不到断气不罢休

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表