Inline ArrayList's array into SmallSortedMap

Store them as Object[], because you can't have a Entry[], because Entry[] has a generic 'this' over <K, V>, you get error "generic array creation" if you try to make "new Entry[]".

And if you try to cast Object[] to Entry[], you get ClassCastException. So I've just made it an object array that we cast when reading out of.

This should save memory and improve performance, because we have fewer pointers to chase.

Also fixed some warnings about unnecessary unchecked suppressions, and type-name should end with T.

PiperOrigin-RevId: 658585983
pull/17681/head
Mark Hansen 8 months ago committed by Copybara-Service
parent 58a97a4b57
commit 910f62779f
  1. 112
      java/core/src/main/java/com/google/protobuf/SmallSortedMap.java

@ -9,7 +9,6 @@ package com.google.protobuf;
import java.util.AbstractMap; import java.util.AbstractMap;
import java.util.AbstractSet; import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
@ -64,25 +63,21 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
* Creates a new instance for mapping FieldDescriptors to their values. The {@link * Creates a new instance for mapping FieldDescriptors to their values. The {@link
* #makeImmutable()} implementation will convert the List values of any repeated fields to * #makeImmutable()} implementation will convert the List values of any repeated fields to
* unmodifiable lists. * unmodifiable lists.
*
* @param arraySize The size of the entry array containing the lexicographically smallest
* mappings.
*/ */
static <FieldDescriptorType extends FieldSet.FieldDescriptorLite<FieldDescriptorType>> static <FieldDescriptorT extends FieldSet.FieldDescriptorLite<FieldDescriptorT>>
SmallSortedMap<FieldDescriptorType, Object> newFieldMap() { SmallSortedMap<FieldDescriptorT, Object> newFieldMap() {
return new SmallSortedMap<FieldDescriptorType, Object>() { return new SmallSortedMap<FieldDescriptorT, Object>() {
@Override @Override
@SuppressWarnings("unchecked")
public void makeImmutable() { public void makeImmutable() {
if (!isImmutable()) { if (!isImmutable()) {
for (int i = 0; i < getNumArrayEntries(); i++) { for (int i = 0; i < getNumArrayEntries(); i++) {
final Map.Entry<FieldDescriptorType, Object> entry = getArrayEntryAt(i); final Map.Entry<FieldDescriptorT, Object> entry = getArrayEntryAt(i);
if (entry.getKey().isRepeated()) { if (entry.getKey().isRepeated()) {
final List<?> value = (List) entry.getValue(); final List<?> value = (List) entry.getValue();
entry.setValue(Collections.unmodifiableList(value)); entry.setValue(Collections.unmodifiableList(value));
} }
} }
for (Map.Entry<FieldDescriptorType, Object> entry : getOverflowEntries()) { for (Map.Entry<FieldDescriptorT, Object> entry : getOverflowEntries()) {
if (entry.getKey().isRepeated()) { if (entry.getKey().isRepeated()) {
final List<?> value = (List) entry.getValue(); final List<?> value = (List) entry.getValue();
entry.setValue(Collections.unmodifiableList(value)); entry.setValue(Collections.unmodifiableList(value));
@ -99,10 +94,14 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
return new SmallSortedMap<>(); return new SmallSortedMap<>();
} }
// The "entry array" is actually a List because generic arrays are not // Only has Entry elements inside.
// allowed. ArrayList also nicely handles the entry shifting on inserts and // Can't declare this as Entry[] because Entry is generic, so you get "generic array creation"
// removes. // error. Instead, use an Object[], and cast to Entry on read.
private List<Entry> entryList; // null Object[] means 'empty'.
private Object[] entries;
// Number of elements in entries that are valid, like ArrayList.size.
private int entriesSize;
private Map<K, V> overflowEntries; private Map<K, V> overflowEntries;
private boolean isImmutable; private boolean isImmutable;
// The EntrySet is a stateless view of the Map. It's initialized the first // The EntrySet is a stateless view of the Map. It's initialized the first
@ -112,7 +111,6 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
private volatile DescendingEntrySet lazyDescendingEntrySet; private volatile DescendingEntrySet lazyDescendingEntrySet;
private SmallSortedMap() { private SmallSortedMap() {
this.entryList = Collections.emptyList();
this.overflowEntries = Collections.emptyMap(); this.overflowEntries = Collections.emptyMap();
this.overflowEntriesDescending = Collections.emptyMap(); this.overflowEntriesDescending = Collections.emptyMap();
} }
@ -120,8 +118,8 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
/** Make this map immutable from this point forward. */ /** Make this map immutable from this point forward. */
public void makeImmutable() { public void makeImmutable() {
if (!isImmutable) { if (!isImmutable) {
// Note: There's no need to wrap the entryList in an unmodifiableList // Note: There's no need to wrap the entries in an unmodifiableList
// because none of the list's accessors are exposed. The iterator() of // because none of the array's accessors are exposed. The iterator() of
// overflowEntries, on the other hand, is exposed so it must be made // overflowEntries, on the other hand, is exposed so it must be made
// unmodifiable. // unmodifiable.
overflowEntries = overflowEntries =
@ -143,12 +141,17 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
/** @return The number of entries in the entry array. */ /** @return The number of entries in the entry array. */
public int getNumArrayEntries() { public int getNumArrayEntries() {
return entryList.size(); return entriesSize;
} }
/** @return The array entry at the given {@code index}. */ /** @return The array entry at the given {@code index}. */
public Map.Entry<K, V> getArrayEntryAt(int index) { public Map.Entry<K, V> getArrayEntryAt(int index) {
return entryList.get(index); if (index >= entriesSize) {
throw new ArrayIndexOutOfBoundsException(index);
}
@SuppressWarnings("unchecked")
Entry e = (Entry) entries[index];
return e;
} }
/** @return There number of overflow entries. */ /** @return There number of overflow entries. */
@ -165,7 +168,7 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
@Override @Override
public int size() { public int size() {
return entryList.size() + overflowEntries.size(); return entriesSize + overflowEntries.size();
} }
/** /**
@ -191,7 +194,9 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
final K key = (K) o; final K key = (K) o;
final int index = binarySearchInArray(key); final int index = binarySearchInArray(key);
if (index >= 0) { if (index >= 0) {
return entryList.get(index).getValue(); @SuppressWarnings("unchecked")
Entry e = (Entry) entries[index];
return e.getValue();
} }
return overflowEntries.get(key); return overflowEntries.get(key);
} }
@ -202,7 +207,9 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
final int index = binarySearchInArray(key); final int index = binarySearchInArray(key);
if (index >= 0) { if (index >= 0) {
// Replace existing array entry. // Replace existing array entry.
return entryList.get(index).setValue(value); @SuppressWarnings("unchecked")
Entry e = (Entry) entries[index];
return e.setValue(value);
} }
ensureEntryArrayMutable(); ensureEntryArrayMutable();
final int insertionPoint = -(index + 1); final int insertionPoint = -(index + 1);
@ -211,20 +218,26 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
return getOverflowEntriesMutable().put(key, value); return getOverflowEntriesMutable().put(key, value);
} }
// Insert new Entry in array. // Insert new Entry in array.
if (entryList.size() == DEFAULT_FIELD_MAP_ARRAY_SIZE) { if (entriesSize == DEFAULT_FIELD_MAP_ARRAY_SIZE) {
// Shift the last array entry into overflow. // Shift the last array entry into overflow.
final Entry lastEntryInArray = entryList.remove(DEFAULT_FIELD_MAP_ARRAY_SIZE - 1); @SuppressWarnings("unchecked")
final Entry lastEntryInArray = (Entry) entries[DEFAULT_FIELD_MAP_ARRAY_SIZE - 1];
entriesSize--;
getOverflowEntriesMutable().put(lastEntryInArray.getKey(), lastEntryInArray.getValue()); getOverflowEntriesMutable().put(lastEntryInArray.getKey(), lastEntryInArray.getValue());
} }
entryList.add(insertionPoint, new Entry(key, value)); System.arraycopy(
entries, insertionPoint, entries, insertionPoint + 1, entries.length - insertionPoint - 1);
entries[insertionPoint] = new Entry(key, value);
entriesSize++;
return null; return null;
} }
@Override @Override
public void clear() { public void clear() {
checkMutable(); checkMutable();
if (!entryList.isEmpty()) { if (entriesSize != 0) {
entryList.clear(); entries = null;
entriesSize = 0;
} }
if (!overflowEntries.isEmpty()) { if (!overflowEntries.isEmpty()) {
overflowEntries.clear(); overflowEntries.clear();
@ -256,12 +269,17 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
private V removeArrayEntryAt(int index) { private V removeArrayEntryAt(int index) {
checkMutable(); checkMutable();
final V removed = entryList.remove(index).getValue(); @SuppressWarnings("unchecked")
final V removed = ((Entry) entries[index]).getValue();
// shift items across
System.arraycopy(entries, index + 1, entries, index, entriesSize - index - 1);
entriesSize--;
if (!overflowEntries.isEmpty()) { if (!overflowEntries.isEmpty()) {
// Shift the first entry in the overflow to be the last entry in the // Shift the first entry in the overflow to be the last entry in the
// array. // array.
final Iterator<Map.Entry<K, V>> iterator = getOverflowEntriesMutable().entrySet().iterator(); final Iterator<Map.Entry<K, V>> iterator = getOverflowEntriesMutable().entrySet().iterator();
entryList.add(new Entry(iterator.next())); entries[entriesSize] = new Entry(iterator.next());
entriesSize++;
iterator.remove(); iterator.remove();
} }
return removed; return removed;
@ -274,13 +292,14 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
*/ */
private int binarySearchInArray(K key) { private int binarySearchInArray(K key) {
int left = 0; int left = 0;
int right = entryList.size() - 1; int right = entriesSize - 1;
// Optimization: For the common case in which entries are added in // Optimization: For the common case in which entries are added in
// ascending tag order, check the largest element in the array before // ascending tag order, check the largest element in the array before
// doing a full binary search. // doing a full binary search.
if (right >= 0) { if (right >= 0) {
int cmp = key.compareTo(entryList.get(right).getKey()); @SuppressWarnings("unchecked")
int cmp = key.compareTo(((Entry) entries[right]).getKey());
if (cmp > 0) { if (cmp > 0) {
return -(right + 2); // Insert point is after "right". return -(right + 2); // Insert point is after "right".
} else if (cmp == 0) { } else if (cmp == 0) {
@ -290,7 +309,8 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
while (left <= right) { while (left <= right) {
int mid = (left + right) / 2; int mid = (left + right) / 2;
int cmp = key.compareTo(entryList.get(mid).getKey()); @SuppressWarnings("unchecked")
int cmp = key.compareTo(((Entry) entries[mid]).getKey());
if (cmp < 0) { if (cmp < 0) {
right = mid - 1; right = mid - 1;
} else if (cmp > 0) { } else if (cmp > 0) {
@ -344,11 +364,13 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
return (SortedMap<K, V>) overflowEntries; return (SortedMap<K, V>) overflowEntries;
} }
/** Lazily creates the entry list. Any code that adds to the list must first call this method. */ /**
* Lazily creates the entry array. Any code that adds to the array must first call this method.
*/
private void ensureEntryArrayMutable() { private void ensureEntryArrayMutable() {
checkMutable(); checkMutable();
if (entryList.isEmpty() && !(entryList instanceof ArrayList)) { if (entries == null) {
entryList = new ArrayList<>(DEFAULT_FIELD_MAP_ARRAY_SIZE); entries = new Object[DEFAULT_FIELD_MAP_ARRAY_SIZE];
} }
} }
@ -498,7 +520,7 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
@Override @Override
public boolean hasNext() { public boolean hasNext() {
return (pos + 1) < entryList.size() return (pos + 1) < entriesSize
|| (!overflowEntries.isEmpty() && getOverflowIterator().hasNext()); || (!overflowEntries.isEmpty() && getOverflowIterator().hasNext());
} }
@ -507,8 +529,10 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
nextCalledBeforeRemove = true; nextCalledBeforeRemove = true;
// Always increment pos so that we know whether the last returned value // Always increment pos so that we know whether the last returned value
// was from the array or from overflow. // was from the array or from overflow.
if (++pos < entryList.size()) { if (++pos < entriesSize) {
return entryList.get(pos); @SuppressWarnings("unchecked")
Entry e = (Entry) entries[pos];
return e;
} }
return getOverflowIterator().next(); return getOverflowIterator().next();
} }
@ -521,7 +545,7 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
nextCalledBeforeRemove = false; nextCalledBeforeRemove = false;
checkMutable(); checkMutable();
if (pos < entryList.size()) { if (pos < entriesSize) {
removeArrayEntryAt(pos--); removeArrayEntryAt(pos--);
} else { } else {
getOverflowIterator().remove(); getOverflowIterator().remove();
@ -547,12 +571,12 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
*/ */
private class DescendingEntryIterator implements Iterator<Map.Entry<K, V>> { private class DescendingEntryIterator implements Iterator<Map.Entry<K, V>> {
private int pos = entryList.size(); private int pos = entriesSize;
private Iterator<Map.Entry<K, V>> lazyOverflowIterator; private Iterator<Map.Entry<K, V>> lazyOverflowIterator;
@Override @Override
public boolean hasNext() { public boolean hasNext() {
return (pos > 0 && pos <= entryList.size()) || getOverflowIterator().hasNext(); return (pos > 0 && pos <= entriesSize) || getOverflowIterator().hasNext();
} }
@Override @Override
@ -560,7 +584,9 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
if (getOverflowIterator().hasNext()) { if (getOverflowIterator().hasNext()) {
return getOverflowIterator().next(); return getOverflowIterator().next();
} }
return entryList.get(--pos); @SuppressWarnings("unchecked")
Entry e = (Entry) entries[--pos];
return e;
} }
@Override @Override
@ -621,7 +647,7 @@ class SmallSortedMap<K extends Comparable<K>, V> extends AbstractMap<K, V> {
int h = 0; int h = 0;
final int listSize = getNumArrayEntries(); final int listSize = getNumArrayEntries();
for (int i = 0; i < listSize; i++) { for (int i = 0; i < listSize; i++) {
h += entryList.get(i).hashCode(); h += entries[i].hashCode();
} }
// Avoid the iterator allocation if possible. // Avoid the iterator allocation if possible.
if (getNumOverflowEntries() > 0) { if (getNumOverflowEntries() > 0) {

Loading…
Cancel
Save