All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.kiwiproject.collect.KiwiLists Maven / Gradle / Ivy

Go to download

Kiwi is a utility library. We really like Google's Guava, and also use Apache Commons. But if they don't have something we need, and we think it is useful, this is where we put it.

The newest version!
package org.kiwiproject.collect;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static java.util.Objects.nonNull;

import lombok.experimental.UtilityClass;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Optional;
import java.util.stream.IntStream;

/**
 * Utility methods for working with {@link List} instances.
 */
@UtilityClass
public class KiwiLists {

    /**
     * Checks whether the specified list is null or empty.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return {@code true} if {@code list} is null or empty; {@code false} otherwise
     */
    public static  boolean isNullOrEmpty(List items) {
        return items == null || items.isEmpty();
    }

    /**
     * Checks whether the specified list is neither null nor empty.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return {@code true} if list is NOT null or empty; {@code false} otherwise
     */
    public static  boolean isNotNullOrEmpty(List items) {
        return !isNullOrEmpty(items);
    }

    /**
     * Checks whether the specified list is non-null and has only one item.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return {@code true} if {@code list} is non-null and has exactly one item; {@code false} otherwise
     */
    public static  boolean hasOneElement(List items) {
        return nonNull(items) && items.size() == 1;
    }

    /**
     * Given a list, sort it according to the natural order, returning a new list.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return a new sorted list
     */
    public static  List sorted(List items) {
        checkNonNullInputList(items);
        return items.stream().sorted().toList();
    }

    /**
     * Given a list, sort it according to the provided {@link Comparator} returning a new list.
     *
     * @param items      the list
     * @param comparator a Comparator to be used to compare stream elements
     * @param         the type of items in the list
     * @return a new sorted list
     */
    public static  List sorted(List items, Comparator comparator) {
        checkNonNullInputList(items);
        checkNotNull(comparator, "Comparator cannot be null");
        return items.stream().sorted(comparator).toList();
    }

    /**
     * Return the first element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the first item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least one item
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T first(List items) {
        return nth(items, 1);
    }

    /**
     * Returns an {@link Optional} containing the first element in the specified list of items, or an empty optional
     * if the list is null or empty.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return Optional containing the first element if exists, otherwise Optional.empty()
     */
    public static  Optional firstIfPresent(List items) {
        return isNotNullOrEmpty(items) ? Optional.of(first(items)) : Optional.empty();
    }

    /**
     * Return the second element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the second item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least two items
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T second(List items) {
        return nth(items, 2);
    }

    /**
     * Return the third element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the third item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least three items
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T third(List items) {
        return nth(items, 3);
    }

    /**
     * Return the fourth element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the fourth item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least four items
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T fourth(List items) {
        return nth(items, 4);
    }

    /**
     * Return the fifth element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the fifth item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least five items
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T fifth(List items) {
        return nth(items, 5);
    }

    /**
     * Returns the penultimate (second to last) element in the specified list.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the penultimate item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least two items
     * @throws java.lang.NullPointerException     if the list is null
     * @see #secondToLast(List)
     */
    public static  T penultimate(List items) {
        checkMinimumSize(items, 2);
        return nth(items, items.size() - 1);
    }

    /**
     * Synonym for {@link #penultimate(List)}.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the penultimate item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least two items
     * @throws java.lang.NullPointerException     if the list is null
     * @see #penultimate(List)
     */
    public static  T secondToLast(List items) {
        return penultimate(items);
    }

    /**
     * Returns the last element in the specified list of items.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return the last item in the list
     * @throws java.lang.IllegalArgumentException if the list does not contain at least one item
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T last(List items) {
        return nth(items, items.size());
    }

    /**
     * Returns an {@link Optional} containing the last element in the specified list of items, or an empty optional
     * if the list is null or empty.
     *
     * @param items the list
     * @param    the type of items in the list
     * @return Optional containing last element if exists, otherwise Optional.empty()
     */
    public static  Optional lastIfPresent(List items) {
        return isNotNullOrEmpty(items) ? Optional.of(last(items)) : Optional.empty();
    }

    /**
     * Return the nth element in the specified list of items, starting at one for the first element, two for the
     * second, etc.
     *
     * @param items  the list
     * @param number the number of the element to retrieve, starting at one (not zero)
     * @param     the type of items in the list
     * @return the nth item in items
     * @throws java.lang.IllegalArgumentException if the list does not contain at least number items
     * @throws java.lang.NullPointerException     if the list is null
     */
    public static  T nth(List items, int number) {
        checkMinimumSize(items, number);
        return items.get(number - 1);
    }

    /**
     * Returns a list of the collection elements with duplicates stripped out.
     *
     * @param collection the collection of values
     * @param         the type of items in the collection
     * @return a new list with only unique elements
     * @throws java.lang.IllegalArgumentException if the collection is null
     */
    public static  List distinct(Collection collection) {
        checkArgument(nonNull(collection), "collection can not be null");
        return  distinctListFrom(collection);
    }

    /**
     * Returns a list of the collection elements with duplicates stripped out or `null` if a null value is passed in.
     *
     * @param collection the collection of values
     * @param         the type of items in the collection
     * @return a new list with only unique elements or null.
     */
    public static  List distinctOrNull(Collection collection) {
        return nonNull(collection) ? distinctListFrom(collection) : null;
    }

    /**
     * Returns a list of the collection elements with duplicates stripped out or an empty list if the input collection
     * is null (or empty).
     *
     * @param collection the collection of values
     * @param         the type of items in the collection
     * @return a new list with only unique elements or an empty list
     */
    public static  List distinctOrEmpty(Collection collection) {
        return nonNull(collection) ? distinctListFrom(collection) : new ArrayList<>();
    }

    private static  List distinctListFrom(Collection collection) {
        return collection.stream().distinct().toList();
    }

    /**
     * Returns a new list with the same elements and the same size as the original. However, the initial position in the list
     * is now the element specified by the "startOffset" and the list wraps around through the contents to end with ("startOffset" - 1).
     *
     * @param input       the original list
     * @param startOffset the desired offset to start the new list
     * @param          the type of the items in the list
     * @return a new list starting at the desired offset
     */
    public static  List newListStartingAtCircularOffset(List input, long startOffset) {
        var size = input.size();
        return IntStream.range(0, size).mapToObj(i -> input.get((int) (startOffset + i) % size)).toList();
    }

    /**
     * Returns a view of the portion of the given list excluding the first element.
     * 

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param the type of the items in the list * @return a view of the given list excluding the first item backed by the original list * @throws NullPointerException if the list is null * @see List#subList(int, int) */ public static List subListExcludingFirst(List items) { checkNonNullInputList(items); if (items.isEmpty()) { return zeroSubList(items); } return items.subList(1, items.size()); } /** * Returns a view of the portion of the given list excluding the last element. *

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param the type of the items in the list * @return a view of the given list excluding the last item backed by the original list * @throws NullPointerException if the list is null * @see List#subList(int, int) */ public static List subListExcludingLast(List items) { checkNonNullInputList(items); if (items.isEmpty()) { return zeroSubList(items); } return items.subList(0, items.size() - 1); } private static List zeroSubList(List items) { return items.subList(0, 0); } /** * Returns a view of the portion of the given list starting at the given logical element number, where the * numbers start at one, until and including the last element in the list. This is useful if something is using * one-based element numbers for some reason. Use {@link #subListFromIndex(List, int)} if you want to use zero-based * list indices. *

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param number the number of the element to start the sublist, starting at one (not zero) * @param the type of items in the list * @return a view of the given list backed by the original list, starting at the given one-based number * @throws NullPointerException if the list is null * @throws IllegalArgumentException if the given number is negative or is higher than the size of the list * @see List#subList(int, int) */ public static List subListFrom(List items, int number) { checkMinimumSize(items, number); return items.subList(number - 1, items.size()); } /** * Returns a view of the portion of the given list starting at the given index, until and including the last * element in the list. *

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param index the index in the list to start the sublist, zero-based like normal List methods * @param the type of items in the list * @return a view of the given list backed by the original list, starting at the given zero-based index * @throws NullPointerException if the list is null * @throws IllegalArgumentException if the given index is negative or is higher than the last index in the list * @see List#subList(int, int) */ public static List subListFromIndex(List items, int index) { checkMinimumSize(items, index + 1); return items.subList(index, items.size()); } /** * Returns a view of the "first N" elements from the input list. *

* If the given number is larger than the size of the list, the entire list is returned, rather than throw * an exception. In this case, the input list is returned directly, i.e. {@code return items}. *

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param number the number of items wanted from the start of the list * @param the type of items in the list * @return a view of the given list, backed by the original list, containing the last {@code number} elements * @see List#subList(int, int) */ public static List firstN(List items, int number) { checkNonNullInputList(items); checkMinSizeIsPositive(number); if (number > items.size()) { return items; } return items.subList(0, number); } /** * Returns a view of the "last N" elements from the input list. *

* If the given number is larger than the size of the list, the entire list is returned, rather than throw * an exception. In this case, the input list is returned directly, i.e. {@code return items}. *

* This method has the same semantics as {@link List#subList(int, int)} since it calls that method. * * @param items the list * @param number the number of items wanted from the end of the list * @param the type of items in the list * @return a view of the given list, backed by the original list, containing the first {@code number} elements * @see List#subList(int, int) */ public static List lastN(List items, int number) { checkNonNullInputList(items); checkMinSizeIsPositive(number); if (number > items.size()) { return items; } var startIndex = items.size() - number; return items.subList(startIndex, items.size()); } /** * Checks that the given list is not null and has the given minimum size. * * @param items the list * @param minSize the minimum required size * @param the type of the items in the list * @throws NullPointerException if the list is null * @throws IllegalArgumentException if minSize is not positive or the list does not contain minSize elements */ public static void checkMinimumSize(List items, int minSize) { checkNonNullInputList(items); checkMinSizeIsPositive(minSize); checkArgument(items.size() >= minSize, "expected at least %s items (actual size: %s)", minSize, items.size()); } private static void checkMinSizeIsPositive(int minSize) { checkArgument(minSize > 0, "number must be positive"); } /** * Checks that the given list is not null. * * @param items the list * @param the type of the items in the list * @throws NullPointerException if the list is null */ public static void checkNonNullInputList(List items) { checkNotNull(items, "items cannot be null"); } /** * Create a new unmodifiable {@link List} with the given items shuffled using {@link Collections#shuffle(List)}. * * @param items the items to include in the new list * @param the type of the items in the list * @return an unmodifiable List with the given items in pseudorandom order */ @SafeVarargs public static List shuffledListOf(T... items) { return List.copyOf(shuffledArrayListOf(items)); } /** * Create a new {@link ArrayList} with the given items shuffled using {@link Collections#shuffle(List)}. * * @param items the items to include in the new list * @param the type of the items in the list * @return an ArrayList with the given items in pseudorandom order */ @SafeVarargs public static List shuffledArrayListOf(T... items) { var list = new ArrayList(items.length); Collections.addAll(list, items); Collections.shuffle(list); return list; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy