2015-12-08 04:51:55 +08:00
|
|
|
#pragma once
|
2015-07-01 14:38:03 +08:00
|
|
|
|
|
|
|
#include <memory>
|
2017-02-18 18:54:37 +08:00
|
|
|
#include <vector>
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-01-23 21:56:37 +08:00
|
|
|
template <class uintXX_t = uint32_t>
|
|
|
|
/**
|
|
|
|
* UnionFind data structure. Provides means of connectivity
|
2017-02-21 01:50:59 +08:00
|
|
|
* setting and checking in O(alpha(n)) amortized complexity. Memory
|
2017-01-23 21:56:37 +08:00
|
|
|
* complexity is linear.
|
|
|
|
*/
|
2017-02-18 18:54:37 +08:00
|
|
|
class UnionFind {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Constructor, creates a UnionFind structure of fixed size.
|
|
|
|
*
|
|
|
|
* @param n Number of elements in the data structure.
|
|
|
|
*/
|
2017-02-21 01:50:59 +08:00
|
|
|
UnionFind(uintXX_t n) : set_count(n), rank(n), parent(n) {
|
|
|
|
for (auto i = 0; i < n; ++i) rank[i] = 0, parent[i] = i;
|
2017-02-18 18:54:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connects two elements (and thereby the sets they belong
|
|
|
|
* to). If they are already connected the function has no effect.
|
|
|
|
*
|
2017-02-21 01:50:59 +08:00
|
|
|
* Has O(alpha(n)) amortized time complexity.
|
2017-02-18 18:54:37 +08:00
|
|
|
*
|
|
|
|
* @param p First element.
|
|
|
|
* @param q Second element.
|
|
|
|
*/
|
|
|
|
void connect(uintXX_t p, uintXX_t q) {
|
|
|
|
auto rp = root(p);
|
|
|
|
auto rq = root(q);
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
// if roots are equal, we don't have to do anything
|
|
|
|
if (rp == rq) return;
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-21 01:50:59 +08:00
|
|
|
// merge the subtree with the smaller rank to the root of the subtree with
|
|
|
|
// the larger rank
|
|
|
|
if (rank[rp] < rank[rq])
|
|
|
|
parent[rp] = rq;
|
|
|
|
else if (rank[rp] > rank[rq])
|
|
|
|
parent[rq] = rp;
|
2017-02-18 18:54:37 +08:00
|
|
|
else
|
2017-02-21 01:50:59 +08:00
|
|
|
parent[rq] = rp, rank[rp] += 1;
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
// update the number of groups
|
|
|
|
set_count--;
|
|
|
|
}
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
/**
|
2017-02-21 01:50:59 +08:00
|
|
|
* Indicates if two elements are connected. Has amortized O(alpha(n)) time
|
2017-02-18 18:54:37 +08:00
|
|
|
* complexity.
|
|
|
|
*
|
|
|
|
* @param p First element.
|
|
|
|
* @param q Second element.
|
|
|
|
* @return See above.
|
|
|
|
*/
|
|
|
|
bool find(uintXX_t p, uintXX_t q) { return root(p) == root(q); }
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
/**
|
|
|
|
* Returns the number of disjoint sets in this UnionFind.
|
|
|
|
*
|
|
|
|
* @return See above.
|
|
|
|
*/
|
|
|
|
uintXX_t size() const { return set_count; }
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
private:
|
|
|
|
uintXX_t set_count;
|
2017-01-23 21:56:37 +08:00
|
|
|
|
2017-02-21 01:50:59 +08:00
|
|
|
// array of subtree ranks
|
|
|
|
std::vector<uintXX_t> rank;
|
2017-01-23 21:56:37 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
// array of tree indices
|
|
|
|
std::vector<uintXX_t> parent;
|
2017-01-23 21:56:37 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
uintXX_t root(uintXX_t p) {
|
|
|
|
auto r = p;
|
|
|
|
auto newp = p;
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
// find the node connected to itself, that's the root
|
|
|
|
while (parent[r] != r) r = parent[r];
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
// do some path compression to enable faster searches
|
|
|
|
while (p != r) newp = parent[p], parent[p] = r, p = newp;
|
2015-07-01 14:38:03 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
return r;
|
|
|
|
}
|
2015-07-01 14:38:03 +08:00
|
|
|
};
|