# HG changeset patch # User redestad # Date 1525072544 -7200 # Node ID 96d4658eb7f275a3ddfb2d4d2afcf9953226368f # Parent 8b9c78f0a712615e9888a345bd24cf4fa69e1b86 8201650: Move iteration order randomization of unmodifiable Set and Map to iterators Reviewed-by: smarks, jiangli diff -r 8b9c78f0a712 -r 96d4658eb7f2 src/java.base/share/classes/java/util/ImmutableCollections.java --- a/src/java.base/share/classes/java/util/ImmutableCollections.java Mon Apr 30 11:59:42 2018 +0530 +++ b/src/java.base/share/classes/java/util/ImmutableCollections.java Mon Apr 30 09:15:44 2018 +0200 @@ -288,7 +288,7 @@ * Constructs a sublist of another SubList. */ static SubList fromSubList(SubList parent, int fromIndex, int toIndex) { - return new SubList(parent.root, parent.offset + fromIndex, toIndex - fromIndex); + return new SubList<>(parent.root, parent.offset + fromIndex, toIndex - fromIndex); } /** @@ -296,7 +296,7 @@ * not a SubList itself. */ static SubList fromList(List list, int fromIndex, int toIndex) { - return new SubList(list, fromIndex, toIndex - fromIndex); + return new SubList<>(list, fromIndex, toIndex - fromIndex); } public E get(int index) { @@ -309,12 +309,12 @@ } public Iterator iterator() { - return new ListItr(this, size()); + return new ListItr<>(this, size()); } public ListIterator listIterator(int index) { rangeCheck(index); - return new ListItr(this, size(), index); + return new ListItr<>(this, size(), index); } public List subList(int fromIndex, int toIndex) { @@ -472,13 +472,8 @@ throw new IllegalArgumentException("duplicate element: " + e0); } - if (SALT >= 0) { - this.e0 = e0; - this.e1 = e1; - } else { - this.e0 = e1; - this.e1 = e0; - } + this.e0 = e0; + this.e1 = e1; } @Override @@ -510,10 +505,10 @@ public E next() { if (idx == 1) { idx = 0; - return e0; + return SALT >= 0 || e1 == null ? e0 : e1; } else if (idx == 2) { idx = 1; - return e1; + return SALT >= 0 ? e1 : e0; } else { throw new NoSuchElementException(); } @@ -578,29 +573,55 @@ return size > 0 && probe(o) >= 0; } + private final class SetNIterator implements Iterator { + + private int remaining; + + private int idx; + + SetNIterator() { + remaining = size(); + if (remaining > 0) { + idx = Math.floorMod(SALT, elements.length); + } + } + + @Override + public boolean hasNext() { + return remaining > 0; + } + + private int nextIndex() { + int idx = this.idx; + if (SALT >= 0) { + if (++idx >= elements.length) { + idx = 0; + } + } else { + if (--idx < 0) { + idx = elements.length - 1; + } + } + return this.idx = idx; + } + + @Override + public E next() { + if (hasNext()) { + E element; + // skip null elements + while ((element = elements[nextIndex()]) == null) {} + remaining--; + return element; + } else { + throw new NoSuchElementException(); + } + } + } + @Override public Iterator iterator() { - return new Iterator<>() { - private int idx = 0; - - @Override - public boolean hasNext() { - while (idx < elements.length) { - if (elements[idx] != null) - return true; - idx++; - } - return false; - } - - @Override - public E next() { - if (! hasNext()) { - throw new NoSuchElementException(); - } - return elements[idx++]; - } - }; + return new SetNIterator(); } @Override @@ -619,7 +640,7 @@ // Callers are relying on this method to perform an implicit nullcheck // of pe private int probe(Object pe) { - int idx = Math.floorMod(pe.hashCode() ^ SALT, elements.length); + int idx = Math.floorMod(pe.hashCode(), elements.length); while (true) { E ee = elements[idx]; if (ee == null) { @@ -806,6 +827,53 @@ return size; } + class MapNIterator implements Iterator> { + + private int remaining; + + private int idx; + + MapNIterator() { + remaining = size(); + if (remaining > 0) { + idx = Math.floorMod(SALT, table.length >> 1) << 1; + } + } + + @Override + public boolean hasNext() { + return remaining > 0; + } + + private int nextIndex() { + int idx = this.idx; + if (SALT >= 0) { + if ((idx += 2) >= table.length) { + idx = 0; + } + } else { + if ((idx -= 2) < 0) { + idx = table.length - 2; + } + } + return this.idx = idx; + } + + @Override + public Map.Entry next() { + if (hasNext()) { + while (table[nextIndex()] == null) {} + @SuppressWarnings("unchecked") + Map.Entry e = + new KeyValueHolder<>((K)table[idx], (V)table[idx+1]); + remaining--; + return e; + } else { + throw new NoSuchElementException(); + } + } + } + @Override public Set> entrySet() { return new AbstractSet<>() { @@ -816,32 +884,7 @@ @Override public Iterator> iterator() { - return new Iterator<>() { - int idx = 0; - - @Override - public boolean hasNext() { - while (idx < table.length) { - if (table[idx] != null) - return true; - idx += 2; - } - return false; - } - - @Override - public Map.Entry next() { - if (hasNext()) { - @SuppressWarnings("unchecked") - Map.Entry e = - new KeyValueHolder<>((K)table[idx], (V)table[idx+1]); - idx += 2; - return e; - } else { - throw new NoSuchElementException(); - } - } - }; + return new MapNIterator(); } }; } @@ -851,7 +894,7 @@ // Callers are relying on this method to perform an implicit nullcheck // of pk. private int probe(Object pk) { - int idx = Math.floorMod(pk.hashCode() ^ SALT, table.length >> 1) << 1; + int idx = Math.floorMod(pk.hashCode(), table.length >> 1) << 1; while (true) { @SuppressWarnings("unchecked") K ek = (K)table[idx];