/* * Copyright 2012, Google Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package org.jf.util; import com.google.common.base.Predicate; import com.google.common.collect.ImmutableSortedSet; import com.google.common.collect.Ordering; import com.google.common.primitives.Ints; import javax.annotation.Nonnull; import java.util.*; public class CollectionUtils { public static int listHashCode(@Nonnull Iterable iterable) { int hashCode = 1; for (T item: iterable) { hashCode = hashCode*31 + item.hashCode(); } return hashCode; } public static int lastIndexOf(@Nonnull Iterable iterable, @Nonnull Predicate predicate) { int index = 0; int lastMatchingIndex = -1; for (T item: iterable) { if (predicate.apply(item)) { lastMatchingIndex = index; } index++; } return lastMatchingIndex; } public static > int compareAsList(@Nonnull Collection list1, @Nonnull Collection list2) { int res = Ints.compare(list1.size(), list2.size()); if (res != 0) return res; Iterator elements2 = list2.iterator(); for (T element1: list1) { res = element1.compareTo(elements2.next()); if (res != 0) return res; } return 0; } public static int compareAsIterable(@Nonnull Comparator comparator, @Nonnull Iterable it1, @Nonnull Iterable it2) { Iterator elements2 = it2.iterator(); for (T element1: it1) { T element2; try { element2 = elements2.next(); } catch (NoSuchElementException ex) { return 1; } int res = comparator.compare(element1, element2); if (res != 0) return res; } if (elements2.hasNext()) { return -1; } return 0; } public static > int compareAsIterable(@Nonnull Iterable it1, @Nonnull Iterable it2) { Iterator elements2 = it2.iterator(); for (T element1: it1) { T element2; try { element2 = elements2.next(); } catch (NoSuchElementException ex) { return 1; } int res = element1.compareTo(element2); if (res != 0) return res; } if (elements2.hasNext()) { return -1; } return 0; } public static int compareAsList(@Nonnull Comparator elementComparator, @Nonnull Collection list1, @Nonnull Collection list2) { int res = Ints.compare(list1.size(), list2.size()); if (res != 0) return res; Iterator elements2 = list2.iterator(); for (T element1: list1) { res = elementComparator.compare(element1, elements2.next()); if (res != 0) return res; } return 0; } @Nonnull public static Comparator> listComparator( @Nonnull final Comparator elementComparator) { return new Comparator>() { @Override public int compare(Collection list1, Collection list2) { return compareAsList(elementComparator, list1, list2); } }; } public static boolean isNaturalSortedSet(@Nonnull Iterable it) { if (it instanceof SortedSet) { SortedSet sortedSet = (SortedSet)it; Comparator comparator = sortedSet.comparator(); return (comparator == null) || comparator.equals(Ordering.natural()); } return false; } public static boolean isSortedSet(@Nonnull Comparator elementComparator, @Nonnull Iterable it) { if (it instanceof SortedSet) { SortedSet sortedSet = (SortedSet)it; Comparator comparator = sortedSet.comparator(); if (comparator == null) { return elementComparator.equals(Ordering.natural()); } return elementComparator.equals(comparator); } return false; } @Nonnull private static SortedSet toNaturalSortedSet(@Nonnull Collection collection) { if (isNaturalSortedSet(collection)) { return (SortedSet)collection; } return ImmutableSortedSet.copyOf(collection); } @Nonnull private static SortedSet toSortedSet(@Nonnull Comparator elementComparator, @Nonnull Collection collection) { if (collection instanceof SortedSet) { SortedSet sortedSet = (SortedSet)collection; Comparator comparator = sortedSet.comparator(); if (comparator != null && comparator.equals(elementComparator)) { return sortedSet; } } return ImmutableSortedSet.copyOf(elementComparator, collection); } @Nonnull public static Comparator> setComparator( @Nonnull final Comparator elementComparator) { return new Comparator>() { @Override public int compare(Collection list1, Collection list2) { return compareAsSet(elementComparator, list1, list2); } }; } public static > int compareAsSet(@Nonnull Collection set1, @Nonnull Collection set2) { int res = Ints.compare(set1.size(), set2.size()); if (res != 0) return res; SortedSet sortedSet1 = toNaturalSortedSet(set1); SortedSet sortedSet2 = toNaturalSortedSet(set2); Iterator elements2 = set2.iterator(); for (T element1: set1) { res = element1.compareTo(elements2.next()); if (res != 0) return res; } return 0; } public static int compareAsSet(@Nonnull Comparator elementComparator, @Nonnull Collection list1, @Nonnull Collection list2) { int res = Ints.compare(list1.size(), list2.size()); if (res != 0) return res; SortedSet set1 = toSortedSet(elementComparator, list1); SortedSet set2 = toSortedSet(elementComparator, list2); Iterator elements2 = set2.iterator(); for (T element1: set1) { res = elementComparator.compare(element1, elements2.next()); if (res != 0) return res; } return 0; } }