This documentation is automatically generated by online-judge-tools/verification-helper
#include "DataStructure/rangeparalleluf.hpp"
#pragma once
#include "DataStructure/unionfind.hpp"
struct RangeParallelUF {
vector<UnionFind> buf;
RangeParallelUF(int n) {
int lg = topbit(n) + 2;
buf.resize(lg);
rep(i, 0, lg) buf[i] = UnionFind(n);
}
template <typename F> void unite(int x, int y, int len, F f) {
if (len == 0)
return;
if (len == 1) {
inner(x, y, 0, f);
return;
}
int k = topbit(len - 1);
inner(x, y, k, f);
inner(x + len - (1 << k), y + len - (1 << k), k, f);
}
private:
template <typename F> void inner(int x, int y, int k, F f) {
if (k == 0) {
int a = buf[0].root(x);
int b = buf[0].root(y);
if (a == b)
return;
buf[0].unite(a, b);
int c = buf[0].root(x);
f(c, a ^ b ^ c);
return;
}
if (!buf[k].unite(x, y))
return;
inner(x, y, k - 1, f);
inner(x + (1 << (k - 1)), y + (1 << (k - 1)), k - 1, f);
}
};
/**
* @brief Range Parallel UnionFind
*/
#line 2 "DataStructure/unionfind.hpp"
struct UnionFind{
vector<int> par; int n;
UnionFind(){}
UnionFind(int _n):par(_n,-1),n(_n){}
int root(int x){return par[x]<0?x:par[x]=root(par[x]);}
bool same(int x,int y){return root(x)==root(y);}
int size(int x){return -par[root(x)];}
bool unite(int x,int y){
x=root(x),y=root(y); if(x==y)return false;
if(size(x)>size(y))swap(x,y);
par[y]+=par[x]; par[x]=y; n--; return true;
}
};
/**
* @brief Union Find
*/
#line 3 "DataStructure/rangeparalleluf.hpp"
struct RangeParallelUF {
vector<UnionFind> buf;
RangeParallelUF(int n) {
int lg = topbit(n) + 2;
buf.resize(lg);
rep(i, 0, lg) buf[i] = UnionFind(n);
}
template <typename F> void unite(int x, int y, int len, F f) {
if (len == 0)
return;
if (len == 1) {
inner(x, y, 0, f);
return;
}
int k = topbit(len - 1);
inner(x, y, k, f);
inner(x + len - (1 << k), y + len - (1 << k), k, f);
}
private:
template <typename F> void inner(int x, int y, int k, F f) {
if (k == 0) {
int a = buf[0].root(x);
int b = buf[0].root(y);
if (a == b)
return;
buf[0].unite(a, b);
int c = buf[0].root(x);
f(c, a ^ b ^ c);
return;
}
if (!buf[k].unite(x, y))
return;
inner(x, y, k - 1, f);
inner(x + (1 << (k - 1)), y + (1 << (k - 1)), k - 1, f);
}
};
/**
* @brief Range Parallel UnionFind
*/