pascal.taie.util.collection.UnionFindSet Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of tai-e Show documentation
Show all versions of tai-e Show documentation
An easy-to-learn/use static analysis framework for Java
The newest version!
/*
* Tai-e: A Static Analysis Framework for Java
*
* Copyright (C) 2022 Tian Tan
* Copyright (C) 2022 Yue Li
*
* This file is part of Tai-e.
*
* Tai-e is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation, either version 3
* of the License, or (at your option) any later version.
*
* Tai-e is distributed in the hope that it will be useful,but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
* Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with Tai-e. If not, see .
*/
package pascal.taie.util.collection;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
public class UnionFindSet {
/**
* Associates elements to UF set entries.
*/
private final Map entries = new HashMap<>();
/**
* Number of disjoint sets.
*/
private int setCount;
public UnionFindSet(Collection elems) {
elems.forEach(elem -> entries.put(elem, new Entry(elem)));
setCount = entries.size();
}
/**
* Unions the sets which e1 and e2 belong to, respectively.
*
* @return {@code true} if this union-find set changed as a result
* of this call.
*/
public boolean union(E e1, E e2) {
Entry root1 = findRootEntry(entries.get(e1));
Entry root2 = findRootEntry(entries.get(e2));
if (root1 == root2) {
return false;
} else { // union by rank
if (root1.rank < root2.rank) {
root1.parent = root2;
} else if (root1.rank > root2.rank) {
root2.parent = root1;
} else {
root2.parent = root1;
++root2.rank;
}
--setCount;
return true;
}
}
/**
* @return {@code true} if e1 and e2 belong to the same set.
*/
public boolean isConnected(E e1, E e2) {
Entry root1 = findRootEntry(entries.get(e1));
Entry root2 = findRootEntry(entries.get(e2));
return root1 == root2;
}
/**
* @return the root element of the set which e belongs to.
*/
public E findRoot(E e) {
return findRootEntry(entries.get(e)).elem;
}
/**
* @return number of disjoint sets in this union-find set.
*/
public int numberOfSets() {
return setCount;
}
/**
* @return a collection of all disjoint sets in this union-find set.
*/
public Collection> getDisjointSets() {
return entries.keySet()
.stream()
.collect(Collectors.groupingBy(this::findRoot, Collectors.toSet()))
.values();
}
private Entry findRootEntry(Entry ent) {
if (ent.parent != ent) { // path compression
ent.parent = findRootEntry(ent.parent);
}
return ent.parent;
}
private class Entry {
private final E elem;
private Entry parent;
private int rank;
private Entry(E elem) {
this.elem = elem;
this.parent = this;
this.rank = 0;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy