当前位置:   article > 正文

集合问题(并查集)

集合问题(并查集)

本题链接:登录—专业IT笔试面试备考平台_牛客网

题目:

样例1:

输入
  1. 4 5 9
  2. 2 3 4 5

输出
YES
0 0 1 1

 样例2:

输入
  1. 3 3 4
  2. 1 2 4

输出
NO

思路:

        这道题关键点在于。

        当集合中有一个元素均存在于集合 A 和集合 B 的时候是 NO。

        并且 P_{i} 的范围是 1 ~ 1e9 所以,当 P_{i} >= max(a,b) 的时候也是 NO。

        我们同时可以指定一个 元素范围外的 一个元素作为 根元素集合 A,B

        其次,我们可以将 下标 作为对应的每一个元素,最后进行合并求结果即可。

代码详解如下:

  1. #include <iostream>
  2. #include <vector>
  3. #include <unordered_map>
  4. #define umap unordered_map
  5. #define int long long
  6. #define endl '\n'
  7. #define IOS ios::sync_with_stdio(false),cin.tie(0),cout.tie(0)
  8. using namespace std;
  9. umap<int,int>pos; // 存储元素对应的下标
  10. // 存储元素集合,至于为什么也用 umap ,由于 Pi 的数据范围上限是 1e9
  11. // 我们要将数组无法开辟这么大,所以我们只能弄个映射 来存储对应的 A,B 根元素
  12. umap<int,int>father;
  13. // 并查集查找函数
  14. inline int Finds(int x)
  15. {
  16. int t = x; // 记录其实查找结点
  17. while(x != father[x]) x = father[x]; // 开始查找
  18. father[t] = x; // 路径压缩查找
  19. return x; // 返回结果
  20. }
  21. // 并查集合并操作
  22. inline void Union(int a,int b)
  23. {
  24. a = Finds(a),b = Finds(b); // 查找对应根节点
  25. father[a] = b; // 合并对应根节点
  26. }
  27. inline void solve()
  28. {
  29. int n,a,b;
  30. cin >> n >> a >> b;
  31. int maxs = max(a,b); // 获取对应 a b 最大值
  32. int A = maxs + 1; // 根据对应的最大值,赋值一个元素范围外的元素作为 集合 A 的根节点
  33. int B = maxs + 2; // 根据对应的最大值,赋值一个元素范围外的元素并且不同于集合A的根元素的元素作为 集合 B 的根节点
  34. father[A] = A,father[B] = B; // 集合根节点初始化
  35. vector<int>v(n + 2,0); // 存储对应元素
  36. for(int i = 1;i <= n;++i)
  37. {
  38. cin >> v[i];
  39. if(v[i] >= maxs) // 如果存在 元素 大于 a 和 b ,那么放不了 任意集合,无解输出 NO
  40. {
  41. cout << "NO" << endl;
  42. return ;
  43. }
  44. pos[v[i]] = i; // 映射对应的下标
  45. father[i] = i; // 对应下标 根节点初始化
  46. }
  47. for(int i = 1;i <= n;++i)
  48. {
  49. // 如果对应的元素存在的话,我们将其元素的下标与当前的下标进行操作合并对应的集合
  50. if(pos[b - v[i]]) Union(i,pos[b - v[i]]); // 另一元素存在 集合 b 那么我们合并对应下标
  51. else Union(A,i); //如果不符合那么合并另一个集合
  52. if(pos[a - v[i]]) Union(i,pos[a - v[i]]); // 另一元素存在 集合 a 那么我们合并对应下标
  53. else Union(B,i); //如果不符合那么合并另一个集合
  54. }
  55. A = Finds(A),B = Finds(B); // 根据对应的 结合 根节点元素查找;
  56. if(A == B) cout << "NO" << endl; // 如果最终集合 A 和 集合 B 的根节点也给合并了,说明无解 NO
  57. else
  58. {
  59. cout << "YES" << endl;
  60. for(int i = 1;i <= n;++i)
  61. {
  62. // cout << bool(Finds(i) == B) << ' '; // 这样输出是错误的,有可能这里没考虑一个情况,就是 A == B 的时候,也有可能返回值的原因
  63. if(Finds(i) == A) cout << "0 ";
  64. else cout << "1 ";
  65. }
  66. cout << endl;
  67. }
  68. }
  69. signed main()
  70. {
  71. IOS;
  72. int ___t = 1;
  73. while(___t--) solve();
  74. return 0;
  75. }

最后提交:

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/57226?site
推荐阅读
相关标签
  

闽ICP备14008679号