001/*
002 * Copyright (C) 2016 The Guava Authors
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 * http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016
017package com.google.common.graph;
018
019import static com.google.common.base.Preconditions.checkNotNull;
020import static com.google.common.base.Preconditions.checkState;
021
022import com.google.common.annotations.Beta;
023import com.google.common.base.MoreObjects;
024import com.google.common.base.MoreObjects.ToStringHelper;
025import com.google.common.base.Objects;
026import com.google.common.collect.Maps;
027import com.google.common.collect.Ordering;
028import com.google.errorprone.annotations.Immutable;
029import java.util.Comparator;
030import java.util.Map;
031import org.checkerframework.checker.nullness.compatqual.NullableDecl;
032
033/**
034 * Used to represent the order of elements in a data structure that supports different options for
035 * iteration order guarantees.
036 *
037 * <p>Example usage:
038 *
039 * <pre>{@code
040 * MutableGraph<Integer> graph =
041 *     GraphBuilder.directed().nodeOrder(ElementOrder.<Integer>natural()).build();
042 * }</pre>
043 *
044 * @author Joshua O'Madadhain
045 * @since 20.0
046 */
047@Beta
048@Immutable
049public final class ElementOrder<T> {
050  private final Type type;
051
052  @SuppressWarnings("Immutable") // Hopefully the comparator provided is immutable!
053  @NullableDecl
054  private final Comparator<T> comparator;
055
056  /**
057   * The type of ordering that this object specifies.
058   *
059   * <ul>
060   *   <li>UNORDERED: no order is guaranteed.
061   *   <li>STABLE: ordering is guaranteed to follow a pattern that won't change between releases.
062   *       Some methods may have stronger guarantees.
063   *   <li>INSERTION: insertion ordering is guaranteed.
064   *   <li>SORTED: ordering according to a supplied comparator is guaranteed.
065   * </ul>
066   */
067  public enum Type {
068    UNORDERED,
069    STABLE,
070    INSERTION,
071    SORTED
072  }
073
074  private ElementOrder(Type type, @NullableDecl Comparator<T> comparator) {
075    this.type = checkNotNull(type);
076    this.comparator = comparator;
077    checkState((type == Type.SORTED) == (comparator != null));
078  }
079
080  /** Returns an instance which specifies that no ordering is guaranteed. */
081  public static <S> ElementOrder<S> unordered() {
082    return new ElementOrder<S>(Type.UNORDERED, null);
083  }
084
085  /**
086   * Returns an instance which specifies that ordering is guaranteed to be always be the same across
087   * iterations, and across releases. Some methods may have stronger guarantees.
088   *
089   * <p>This instance is only useful in combination with {@code incidentEdgeOrder}, e.g. {@code
090   * graphBuilder.incidentEdgeOrder(ElementOrder.stable())}.
091   *
092   * <h3>In combination with {@code incidentEdgeOrder}</h3>
093   *
094   * <p>{@code incidentEdgeOrder(ElementOrder.stable())} guarantees the ordering of the returned
095   * collections of the following methods:
096   *
097   * <ul>
098   *   <li>For {@link Graph} and {@link ValueGraph}:
099   *       <ul>
100   *         <li>{@code edges()}: Stable order
101   *         <li>{@code adjacentNodes(node)}: Connecting edge insertion order
102   *         <li>{@code predecessors(node)}: Connecting edge insertion order
103   *         <li>{@code successors(node)}: Connecting edge insertion order
104   *         <li>{@code incidentEdges(node)}: Edge insertion order
105   *       </ul>
106   *   <li>For {@link Network}:
107   *       <ul>
108   *         <li>{@code adjacentNodes(node)}: Stable order
109   *         <li>{@code predecessors(node)}: Connecting edge insertion order
110   *         <li>{@code successors(node)}: Connecting edge insertion order
111   *         <li>{@code incidentEdges(node)}: Stable order
112   *         <li>{@code inEdges(node)}: Edge insertion order
113   *         <li>{@code outEdges(node)}: Edge insertion order
114   *         <li>{@code adjacentEdges(edge)}: Stable order
115   *         <li>{@code edgesConnecting(nodeU, nodeV)}: Edge insertion order
116   *       </ul>
117   * </ul>
118   */
119  // TODO(b/142723300): Make this method public
120  static <S> ElementOrder<S> stable() {
121    return new ElementOrder<S>(Type.STABLE, null);
122  }
123
124  /** Returns an instance which specifies that insertion ordering is guaranteed. */
125  public static <S> ElementOrder<S> insertion() {
126    return new ElementOrder<S>(Type.INSERTION, null);
127  }
128
129  /**
130   * Returns an instance which specifies that the natural ordering of the elements is guaranteed.
131   */
132  public static <S extends Comparable<? super S>> ElementOrder<S> natural() {
133    return new ElementOrder<S>(Type.SORTED, Ordering.<S>natural());
134  }
135
136  /**
137   * Returns an instance which specifies that the ordering of the elements is guaranteed to be
138   * determined by {@code comparator}.
139   */
140  public static <S> ElementOrder<S> sorted(Comparator<S> comparator) {
141    return new ElementOrder<S>(Type.SORTED, comparator);
142  }
143
144  /** Returns the type of ordering used. */
145  public Type type() {
146    return type;
147  }
148
149  /**
150   * Returns the {@link Comparator} used.
151   *
152   * @throws UnsupportedOperationException if comparator is not defined
153   */
154  public Comparator<T> comparator() {
155    if (comparator != null) {
156      return comparator;
157    }
158    throw new UnsupportedOperationException("This ordering does not define a comparator.");
159  }
160
161  @Override
162  public boolean equals(@NullableDecl Object obj) {
163    if (obj == this) {
164      return true;
165    }
166    if (!(obj instanceof ElementOrder)) {
167      return false;
168    }
169
170    ElementOrder<?> other = (ElementOrder<?>) obj;
171    return (type == other.type) && Objects.equal(comparator, other.comparator);
172  }
173
174  @Override
175  public int hashCode() {
176    return Objects.hashCode(type, comparator);
177  }
178
179  @Override
180  public String toString() {
181    ToStringHelper helper = MoreObjects.toStringHelper(this).add("type", type);
182    if (comparator != null) {
183      helper.add("comparator", comparator);
184    }
185    return helper.toString();
186  }
187
188  /** Returns an empty mutable map whose keys will respect this {@link ElementOrder}. */
189  <K extends T, V> Map<K, V> createMap(int expectedSize) {
190    switch (type) {
191      case UNORDERED:
192        return Maps.newHashMapWithExpectedSize(expectedSize);
193      case INSERTION:
194      case STABLE:
195        return Maps.newLinkedHashMapWithExpectedSize(expectedSize);
196      case SORTED:
197        return Maps.newTreeMap(comparator());
198      default:
199        throw new AssertionError();
200    }
201  }
202
203  @SuppressWarnings("unchecked")
204  <T1 extends T> ElementOrder<T1> cast() {
205    return (ElementOrder<T1>) this;
206  }
207}