/* * ============================================================================= * * Copyright (c) 2010, The JAVATUPLES team (http://www.javatuples.org) * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * ============================================================================= */ package de.sanandrew.core.manpack.util.javatuples; import de.sanandrew.core.manpack.util.javatuples.valueintf.IValue0; import de.sanandrew.core.manpack.util.javatuples.valueintf.IValue1; import java.util.ArrayList; import java.util.Collection; import java.util.Iterator; /** * <p> * A tuple of two elements. * </p> * * @author Daniel Fernández * @since 1.0 */ public final class Pair<A, B> extends Tuple implements IValue0<A>, IValue1<B> { private static final long serialVersionUID = 2438099850625502138L; private static final int SIZE = 2; private final A val0; private final B val1; public static <A, B> Pair<A, B> with(final A value0, final B value1) { return new Pair<>(value0, value1); } //region fromXYZ /** * <p> * Create tuple from array. Array has to have exactly two elements. * </p> * * @param <X> the array component type * @param array the array to be converted to a tuple * @return the tuple */ public static <X> Pair<X, X> fromArray(final X[] array) { if( array == null ) { throw new IllegalArgumentException("Array cannot be null"); } if( array.length != 2 ) { throw new IllegalArgumentException(String.format("Array must have exactly 2 elements in order to create a Pair. Size is %d", array.length)); } return new Pair<>(array[0], array[1]); } /** * <p> * Create tuple from collection. Collection has to have exactly two elements. * </p> * * @param <X> the collection component type * @param collection the collection to be converted to a tuple * @return the tuple */ public static <X> Pair<X, X> fromCollection(final Collection<X> collection) { return fromIterable(collection); } /** * <p> * Create tuple from iterable. Iterable has to have exactly two elements. * </p> * * @param <X> the iterable component type * @param iterable the iterable to be converted to a tuple * @return the tuple */ public static <X> Pair<X, X> fromIterable(final Iterable<X> iterable) { return fromIterable(iterable, 0, true); } /** * <p> * Create tuple from iterable, starting from the specified index. Iterable * can have more (or less) elements than the tuple to be created. * </p> * * @param <X> the iterable component type * @param iterable the iterable to be converted to a tuple * @return the tuple */ public static <X> Pair<X, X> fromIterable(final Iterable<X> iterable, int index) { return fromIterable(iterable, index, false); } private static <X> Pair<X, X> fromIterable(final Iterable<X> iterable, int index, final boolean checkSize) { if( iterable == null ) { throw new IllegalArgumentException("Iterable cannot be null"); } X element; ArrayList<X> elements = new ArrayList<>(2); final Iterator<X> iter = iterable.iterator(); int lastIndex = index + SIZE - 1; for( int i = 0; i <= lastIndex; i++ ) { if( iter.hasNext() ) { element = iter.next(); if( i >= index ) { if( checkSize && i == lastIndex && iter.hasNext() ) { throw new IllegalArgumentException("Iterable must have exactly 2 elements in order to create a Pair."); } elements.add(element); } } else { if( i < index ) { throw new IllegalArgumentException(String.format("Iterable has not enough elements to grab a value from index %d", index)); } else { throw new IllegalArgumentException(String.format("Not enough elements for creating a Pair (2 needed, %d given)", i)); } } } return new Pair<>(elements.get(0), elements.get(1)); } public Pair(final A value0, final B value1) { super(value0, value1); this.val0 = value0; this.val1 = value1; } @Override public A getValue0() { return this.val0; } @Override public B getValue1() { return this.val1; } @Override public int getSize() { return SIZE; } }