/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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 org.apache.ignite.internal.util.lang; import java.io.Externalizable; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.util.Iterator; import java.util.NoSuchElementException; import org.apache.ignite.internal.util.tostring.GridToStringInclude; import org.apache.ignite.internal.util.typedef.internal.S; import org.jetbrains.annotations.Nullable; /** * Convenience class representing mutable tuple of a single value. * <h2 class="header">Thread Safety</h2> * This class doesn't provide any synchronization for multi-threaded access * and it is responsibility of the user of this class to provide outside * synchronization, if needed. */ public class GridTuple<V> implements Iterable<V>, Cloneable, Externalizable { /** */ private static final long serialVersionUID = 0L; /** The value to wrap. */ @GridToStringInclude private V val; /** * Empty constructor required by {@link Externalizable}. */ public GridTuple() { // No-op. } /** * Constructs mutable object with given value. * * @param val Wrapped value. */ public GridTuple(@Nullable V val) { this.val = val; } /** * Gets value. * * @return Wrapped value. */ @Nullable public V get() { return val; } /** * Sets value. * * @param val Value to set. */ public void set(@Nullable V val) { this.val = val; } /** {@inheritDoc} */ @Override public Iterator<V> iterator() { return new Iterator<V>() { private boolean hasNext = true; @Override public boolean hasNext() { return hasNext; } @Override public V next() { if (!hasNext()) throw new NoSuchElementException(); hasNext = false; return val; } @Override public void remove() { throw new UnsupportedOperationException(); } }; } /** {@inheritDoc} */ @SuppressWarnings({"CloneDoesntDeclareCloneNotSupportedException"}) @Override public Object clone() { try { return super.clone(); } catch (CloneNotSupportedException ignore) { throw new InternalError(); } } /** {@inheritDoc} */ @Override public void writeExternal(ObjectOutput out) throws IOException { out.writeObject(val); } /** {@inheritDoc} */ @SuppressWarnings({"unchecked"}) @Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { val = (V)in.readObject(); } /** {@inheritDoc} */ @Override public int hashCode() { return val == null ? 0 : val.hashCode(); } /** {@inheritDoc} */ @Override public boolean equals(Object obj) { if (obj == this) return true; if (!(obj instanceof GridTuple)) return false; GridTuple t = (GridTuple)obj; // Both nulls or equals. return val == null ? t.val == null : val.equals(t.val); } /** {@inheritDoc} */ @Override public String toString() { return S.toString(GridTuple.class, this); } }