|
7 | 7 | import ca.spottedleaf.concurrentutil.util.ThrowUtil;
|
8 | 8 | import ca.spottedleaf.concurrentutil.util.Validate;
|
9 | 9 | import java.lang.invoke.VarHandle;
|
| 10 | +import java.util.ArrayList; |
10 | 11 | import java.util.Arrays;
|
| 12 | +import java.util.Collection; |
11 | 13 | import java.util.Iterator;
|
| 14 | +import java.util.List; |
12 | 15 | import java.util.NoSuchElementException;
|
13 | 16 | import java.util.PrimitiveIterator;
|
| 17 | +import java.util.Set; |
14 | 18 | import java.util.concurrent.atomic.LongAdder;
|
15 | 19 | import java.util.function.BiFunction;
|
16 | 20 | import java.util.function.Consumer;
|
@@ -83,6 +87,9 @@ protected final int compareExchangeThresholdVolatile(final int expect, final int
|
83 | 87 | return (int)THRESHOLD_HANDLE.compareAndExchange(this, expect, update);
|
84 | 88 | }
|
85 | 89 |
|
| 90 | + protected Values<V> values; |
| 91 | + protected EntrySet<V> entrySet; |
| 92 | + |
86 | 93 | public ConcurrentLong2ReferenceChainedHashTable() {
|
87 | 94 | this(DEFAULT_CAPACITY, DEFAULT_LOAD_FACTOR);
|
88 | 95 | }
|
@@ -1322,6 +1329,22 @@ public Iterator<V> valueIterator() {
|
1322 | 1329 | return new ValueIterator<>(this);
|
1323 | 1330 | }
|
1324 | 1331 |
|
| 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 | + |
1325 | 1348 | protected static final class EntryIterator<V> extends BaseIteratorImpl<V, TableEntry<V>> {
|
1326 | 1349 |
|
1327 | 1350 | public EntryIterator(final ConcurrentLong2ReferenceChainedHashTable<V> map) {
|
@@ -1618,6 +1641,136 @@ public ResizeChain(final TableEntry<V>[] table, final ResizeChain<V> prev, final
|
1618 | 1641 | }
|
1619 | 1642 | }
|
1620 | 1643 |
|
| 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 | + |
1621 | 1774 | public static final class TableEntry<V> {
|
1622 | 1775 |
|
1623 | 1776 | private static final VarHandle TABLE_ENTRY_ARRAY_HANDLE = ConcurrentUtil.getArrayHandle(TableEntry[].class);
|
|
0 commit comments