cn.nukkit.utils.collection.WeakConcurrentSet Maven / Gradle / Ivy
/*
* Originally from Mockito project.
* ======================================
* Copyright (c) 2016 Mockito contributors
* This program is made available under the terms of the MIT License.
*/
package cn.nukkit.utils.collection;
import org.jetbrains.annotations.NotNull;
import java.lang.ref.Reference;
import java.util.Iterator;
import java.util.Map;
import java.util.function.Consumer;
/**
*
* A thread-safe set with weak values. Entries are based on a key's system hash code and keys are considered equal only by reference equality.
*
* This class does not implement the {@link java.util.Set} interface because this implementation is incompatible
* with the set contract. While iterating over a set's entries, any value that has not passed iteration is referenced non-weakly.
*/
public class WeakConcurrentSet implements Runnable, Iterable {
final WeakConcurrentMap target;
final int parallelismThreshold;
public WeakConcurrentSet(Cleaner cleaner) {
switch (cleaner) {
case INLINE -> target = new WeakConcurrentMap.WithInlinedExpunction<>();
case THREAD, MANUAL -> target = new WeakConcurrentMap<>(cleaner == Cleaner.THREAD);
default -> throw new AssertionError();
}
this.parallelismThreshold = Runtime.getRuntime().availableProcessors();
}
/**
* @param value The value to add to the set.
* @return {@code true} if the value was added to the set and was not contained before.
*/
public boolean add(V value) {
return target.put(value, Boolean.TRUE) == null; // is null or Boolean.TRUE
}
/**
* @param value The value to check if it is contained in the set.
* @return {@code true} if the set contains the value.
*/
public boolean contains(V value) {
return target.containsKey(value);
}
/**
* @param value The value to remove from the set.
* @return {@code true} if the value is contained in the set.
*/
public boolean remove(V value) {
return target.remove(value) != null;
}
/**
* Clears the set.
*/
public void clear() {
target.clear();
}
/**
* Returns the approximate size of this set where the returned number is at least as big as the actual number of entries.
*
* @return The minimum size of this set.
*/
public int approximateSize() {
return target.approximateSize();
}
@Override
public void run() {
target.run();
}
/**
* Determines the cleaning format. A reference is removed either by an explicitly started cleaner thread
* associated with this instance ({@link Cleaner#THREAD}), as a result of interacting with this thread local
* from any thread ({@link Cleaner#INLINE} or manually by submitting the detached thread local to a thread
* ({@link Cleaner#MANUAL}).
*/
public enum Cleaner {
THREAD,
INLINE,
MANUAL
}
/**
* Cleans all unused references.
*/
public void expungeStaleEntries() {
target.expungeStaleEntries();
}
/**
* @return The cleaner thread or {@code null} if no such thread was set.
*/
public Thread getCleanerThread() {
return target.getCleanerThread();
}
@Override
public Iterator iterator() {
return new ReducingIterator(target.iterator());
}
public void parallelForeach(@NotNull Consumer super V> action) {
target.target.forEachKey(parallelismThreshold, Reference::get, action);
}
public void clearDeadReferences() {
target.clearDeadReferences();
}
private static class ReducingIterator implements Iterator {
private final Iterator> iterator;
private ReducingIterator(Iterator> iterator) {
this.iterator = iterator;
}
@Override
public void remove() {
iterator.remove();
}
@Override
public V next() {
return iterator.next().getKey();
}
@Override
public boolean hasNext() {
return iterator.hasNext();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy