From da7c0ad961d5ccfa4c53bbb39a61ed602c39201b Mon Sep 17 00:00:00 2001 From: Francesco Nigro Date: Fri, 30 Aug 2024 21:53:14 +0200 Subject: [PATCH] Use JDK immutable lists to reduce arity of List invokeInterface --- .../java/io/quarkus/qute/ImmutableList.java | 295 ++---------------- .../io/quarkus/qute/ImmutableListTest.java | 4 +- 2 files changed, 21 insertions(+), 278 deletions(-) diff --git a/independent-projects/qute/core/src/main/java/io/quarkus/qute/ImmutableList.java b/independent-projects/qute/core/src/main/java/io/quarkus/qute/ImmutableList.java index 5da8257833c9f..8a2debed07278 100644 --- a/independent-projects/qute/core/src/main/java/io/quarkus/qute/ImmutableList.java +++ b/independent-projects/qute/core/src/main/java/io/quarkus/qute/ImmutableList.java @@ -1,16 +1,8 @@ package io.quarkus.qute; -import java.util.AbstractList; import java.util.ArrayList; -import java.util.Arrays; import java.util.Collection; -import java.util.Collections; -import java.util.Iterator; import java.util.List; -import java.util.ListIterator; -import java.util.NoSuchElementException; -import java.util.Spliterator; -import java.util.Spliterators; /** * Immutable lists. @@ -26,14 +18,18 @@ private ImmutableList() { * @return an immutable copy of the given list */ public static List copyOf(List list) { - if (list.isEmpty()) { - return Collections.emptyList(); - } else if (list.size() == 1) { - return of(list.get(0)); - } else if (list.size() == 2) { - return of(list.get(0), list.get(1)); - } - return new ImmutableArrayList<>(list.toArray()); + final int size; + if (list instanceof ArrayList && (size = list.size()) <= 2) { + switch (size) { + case 0: + return List.of(); + case 1: + return List.of(list.get(0)); + case 2: + return List.of(list.get(0), list.get(1)); + } + } + return List.copyOf(list); } /** @@ -43,16 +39,12 @@ public static List copyOf(List list) { */ @SafeVarargs public static List of(T... elements) { - switch (elements.length) { - case 0: - return Collections.emptyList(); - case 1: - return of(elements[0]); - case 2: - return of(elements[0], elements[1]); - default: - return new ImmutableArrayList<>(elements); - } + return switch (elements.length) { + case 0 -> List.of(); + case 1 -> List.of(elements[0]); + case 2 -> List.of(elements[0], elements[1]); + default -> List.of(elements); + }; } /** @@ -62,7 +54,7 @@ public static List of(T... elements) { * @return an immutable list */ public static List of(E element) { - return Collections.singletonList(element); + return List.of(element); } /** @@ -73,7 +65,7 @@ public static List of(E element) { * @return an immutable list */ public static List of(E e1, E e2) { - return new ImmutableList2<>(e1, e2); + return List.of(e1, e2); } /** @@ -105,253 +97,6 @@ public Builder addAll(Collection elements) { public List build() { return copyOf(elements); } - - } - - static class ImmutableList2 extends AbstractList { - - private final E e0; - private final E e1; - - ImmutableList2(E e0, E e1) { - this.e0 = e0; - this.e1 = e1; - } - - @Override - public E get(int index) { - if (index == 0) { - return e0; - } else if (index == 1) { - return e1; - } - throw indexOutOfBound(index, 2); - } - - @Override - public int size() { - return 2; - } - - @Override - public boolean isEmpty() { - return false; - } - - @Override - public Iterator iterator() { - return new Itr(); - } - - @Override - public ListIterator listIterator() { - return new Itr(); - } - - private final class Itr implements ListIterator { - - private byte cursor = 0; - - @Override - public boolean hasNext() { - return cursor < 2; - } - - @Override - public E next() { - if (cursor == 0) { - cursor++; - return e0; - } else if (cursor == 1) { - cursor++; - return e1; - } - throw new NoSuchElementException(); - } - - @Override - public boolean hasPrevious() { - return cursor > 0; - } - - @Override - public E previous() { - if (cursor == 2) { - cursor--; - return e1; - } else if (cursor == 1) { - cursor--; - return e0; - } - throw new NoSuchElementException(); - } - - @Override - public int nextIndex() { - return cursor; - } - - @Override - public int previousIndex() { - return cursor - 1; - } - - @Override - public void remove() { - throw new UnsupportedOperationException(); - } - - @Override - public void set(E e) { - throw new UnsupportedOperationException(); - } - - @Override - public void add(E e) { - throw new UnsupportedOperationException(); - } - } - - } - - static class ImmutableArrayList extends AbstractList { - - private final Object[] elements; - - ImmutableArrayList(Object[] elements) { - this.elements = elements; - } - - @SuppressWarnings("unchecked") - @Override - public E get(int index) { - if (index < 0 || index >= elements.length) { - throw indexOutOfBound(index, size()); - } - return (E) elements[index]; - } - - @Override - public int size() { - return elements.length; - } - - @Override - public Iterator iterator() { - return new Itr(elements.length, 0); - } - - @Override - public ListIterator listIterator() { - return new Itr(elements.length, 0); - } - - @Override - public ListIterator listIterator(int index) { - return new Itr(elements.length, index); - } - - @Override - public List subList(int fromIndex, int toIndex) { - if (fromIndex < 0 || fromIndex > toIndex) { - throw indexOutOfBound(fromIndex, size()); - } - if (toIndex > elements.length) { - throw indexOutOfBound(toIndex, size()); - } - if (fromIndex == toIndex) { - return Collections.emptyList(); - } - return new ImmutableArrayList<>( - Arrays.copyOfRange(this.elements, fromIndex, toIndex)); - } - - @Override - public String toString() { - return Arrays.toString(elements); - } - - @Override - public boolean isEmpty() { - return elements.length == 0; - } - - @Override - public Spliterator spliterator() { - return Spliterators.spliterator(this, Spliterator.ORDERED - | Spliterator.IMMUTABLE | Spliterator.NONNULL); - } - - private final class Itr implements ListIterator { - - private int cursor; - - private final int size; - - Itr(int size, int position) { - this.size = size; - this.cursor = position; - } - - @Override - public boolean hasNext() { - return cursor < size; - } - - @SuppressWarnings("unchecked") - @Override - public E next() { - if (hasNext()) { - return (E) elements[cursor++]; - } - throw new NoSuchElementException(); - } - - @Override - public boolean hasPrevious() { - return cursor > 0; - } - - @SuppressWarnings("unchecked") - @Override - public E previous() { - if (hasPrevious()) { - return (E) elements[--cursor]; - } - throw new NoSuchElementException(); - } - - @Override - public int nextIndex() { - return cursor; - } - - @Override - public int previousIndex() { - return cursor - 1; - } - - @Override - public void remove() { - throw new UnsupportedOperationException(); - } - - @Override - public void set(E e) { - throw new UnsupportedOperationException(); - } - - @Override - public void add(E e) { - throw new UnsupportedOperationException(); - } - } - - } - - private static IndexOutOfBoundsException indexOutOfBound(int index, int size) { - return new IndexOutOfBoundsException("Index " + index - + " is out of bounds, list size: " + size); } } diff --git a/independent-projects/qute/core/src/test/java/io/quarkus/qute/ImmutableListTest.java b/independent-projects/qute/core/src/test/java/io/quarkus/qute/ImmutableListTest.java index 43d26aa6ae438..2f042def78aeb 100644 --- a/independent-projects/qute/core/src/test/java/io/quarkus/qute/ImmutableListTest.java +++ b/independent-projects/qute/core/src/test/java/io/quarkus/qute/ImmutableListTest.java @@ -43,9 +43,7 @@ public void testListOfTwo() { assertThatExceptionOfType(NoSuchElementException.class) .isThrownBy(() -> it.next()); assertThatExceptionOfType(IndexOutOfBoundsException.class) - .isThrownBy(() -> list.get(5)) - .withMessage( - "Index 5 is out of bounds, list size: 2"); + .isThrownBy(() -> list.get(5)); } }