All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.nukkit.utils.collection.WeakConcurrentSet Maven / Gradle / Ivy

There is a newer version: 1.20.40-r1
Show newest version
/*
 * 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 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