Skip to content

Commit e86113d

Browse files
committed
Add basic values() and entrySet() view for concurrent long map
They do not support removal from the underlying map
1 parent 953555a commit e86113d

File tree

1 file changed

+153
-0
lines changed

1 file changed

+153
-0
lines changed

src/main/java/ca/spottedleaf/concurrentutil/map/ConcurrentLong2ReferenceChainedHashTable.java

+153
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,14 @@
77
import ca.spottedleaf.concurrentutil.util.ThrowUtil;
88
import ca.spottedleaf.concurrentutil.util.Validate;
99
import java.lang.invoke.VarHandle;
10+
import java.util.ArrayList;
1011
import java.util.Arrays;
12+
import java.util.Collection;
1113
import java.util.Iterator;
14+
import java.util.List;
1215
import java.util.NoSuchElementException;
1316
import java.util.PrimitiveIterator;
17+
import java.util.Set;
1418
import java.util.concurrent.atomic.LongAdder;
1519
import java.util.function.BiFunction;
1620
import java.util.function.Consumer;
@@ -83,6 +87,9 @@ protected final int compareExchangeThresholdVolatile(final int expect, final int
8387
return (int)THRESHOLD_HANDLE.compareAndExchange(this, expect, update);
8488
}
8589

90+
protected Values<V> values;
91+
protected EntrySet<V> entrySet;
92+
8693
public ConcurrentLong2ReferenceChainedHashTable() {
8794
this(DEFAULT_CAPACITY, DEFAULT_LOAD_FACTOR);
8895
}
@@ -1322,6 +1329,22 @@ public Iterator<V> valueIterator() {
13221329
return new ValueIterator<>(this);
13231330
}
13241331

1332+
public Collection<V> values() {
1333+
final Values<V> values = this.values;
1334+
if (values != null) {
1335+
return values;
1336+
}
1337+
return this.values = new Values<>(this);
1338+
}
1339+
1340+
public Set<TableEntry<V>> entrySet() {
1341+
final EntrySet<V> entrySet = this.entrySet;
1342+
if (entrySet != null) {
1343+
return entrySet;
1344+
}
1345+
return this.entrySet = new EntrySet<>(this);
1346+
}
1347+
13251348
protected static final class EntryIterator<V> extends BaseIteratorImpl<V, TableEntry<V>> {
13261349

13271350
public EntryIterator(final ConcurrentLong2ReferenceChainedHashTable<V> map) {
@@ -1618,6 +1641,136 @@ public ResizeChain(final TableEntry<V>[] table, final ResizeChain<V> prev, final
16181641
}
16191642
}
16201643

1644+
protected static abstract class BaseCollection<V, E> implements Collection<E> {
1645+
1646+
protected final ConcurrentLong2ReferenceChainedHashTable<V> map;
1647+
1648+
protected BaseCollection(final ConcurrentLong2ReferenceChainedHashTable<V> map) {
1649+
this.map = map;
1650+
}
1651+
1652+
@Override
1653+
public int size() {
1654+
return this.map.size();
1655+
}
1656+
1657+
@Override
1658+
public boolean isEmpty() {
1659+
return this.map.isEmpty();
1660+
}
1661+
1662+
@Override
1663+
public void forEach(final Consumer<? super E> action) {
1664+
this.iterator().forEachRemaining(action);
1665+
}
1666+
1667+
private List<E> asList() {
1668+
final List<E> ret = new ArrayList<>(this.map.size());
1669+
1670+
for (final E element : this) {
1671+
ret.add(element);
1672+
}
1673+
1674+
return ret;
1675+
}
1676+
1677+
@Override
1678+
public Object[] toArray() {
1679+
return this.asList().toArray();
1680+
}
1681+
1682+
@Override
1683+
public <T> T[] toArray(final T[] a) {
1684+
return this.asList().toArray(a);
1685+
}
1686+
1687+
@Override
1688+
public boolean containsAll(final Collection<?> collection) {
1689+
for (final Object value : collection) {
1690+
if (!this.contains(value)) {
1691+
return false;
1692+
}
1693+
}
1694+
1695+
return true;
1696+
}
1697+
1698+
@Override
1699+
public boolean add(final E value) {
1700+
throw new UnsupportedOperationException();
1701+
}
1702+
1703+
@Override
1704+
public boolean remove(final Object value) {
1705+
throw new UnsupportedOperationException();
1706+
}
1707+
1708+
@Override
1709+
public boolean addAll(final Collection<? extends E> collection) {
1710+
throw new UnsupportedOperationException();
1711+
}
1712+
1713+
@Override
1714+
public boolean removeAll(final Collection<?> collection) {
1715+
throw new UnsupportedOperationException();
1716+
}
1717+
1718+
@Override
1719+
public boolean removeIf(final Predicate<? super E> filter) {
1720+
throw new UnsupportedOperationException();
1721+
}
1722+
1723+
@Override
1724+
public boolean retainAll(final Collection<?> collection) {
1725+
throw new UnsupportedOperationException();
1726+
}
1727+
1728+
@Override
1729+
public void clear() {
1730+
throw new UnsupportedOperationException();
1731+
}
1732+
}
1733+
1734+
protected static class Values<V> extends BaseCollection<V, V> {
1735+
1736+
public Values(final ConcurrentLong2ReferenceChainedHashTable<V> map) {
1737+
super(map);
1738+
}
1739+
1740+
@Override
1741+
public boolean contains(final Object value) {
1742+
return this.map.containsValue((V)value);
1743+
}
1744+
1745+
@Override
1746+
public Iterator<V> iterator() {
1747+
return this.map.valueIterator();
1748+
}
1749+
}
1750+
1751+
protected static class EntrySet<V> extends BaseCollection<V, TableEntry<V>> implements Set<TableEntry<V>> {
1752+
1753+
protected EntrySet(final ConcurrentLong2ReferenceChainedHashTable<V> map) {
1754+
super(map);
1755+
}
1756+
1757+
@Override
1758+
public boolean contains(final Object value) {
1759+
if (!(value instanceof ConcurrentLong2ReferenceChainedHashTable.TableEntry<?> entry)) {
1760+
return false;
1761+
}
1762+
1763+
final V mapped = this.map.get(entry.getKey());
1764+
1765+
return mapped != null && mapped == value;
1766+
}
1767+
1768+
@Override
1769+
public Iterator<TableEntry<V>> iterator() {
1770+
return this.map.entryIterator();
1771+
}
1772+
}
1773+
16211774
public static final class TableEntry<V> {
16221775

16231776
private static final VarHandle TABLE_ENTRY_ARRAY_HANDLE = ConcurrentUtil.getArrayHandle(TableEntry[].class);

0 commit comments

Comments
 (0)