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.graph.GraphConstants.GRAPH_STRING_FORMAT;
020import static com.google.common.graph.GraphConstants.MULTIPLE_EDGES_CONNECTING;
021
022import com.google.common.annotations.Beta;
023import com.google.common.base.Function;
024import com.google.common.base.Optional;
025import com.google.common.collect.ImmutableSet;
026import com.google.common.collect.Iterators;
027import com.google.common.collect.Maps;
028import com.google.common.collect.Sets;
029import com.google.common.math.IntMath;
030import java.util.AbstractSet;
031import java.util.Iterator;
032import java.util.Map;
033import java.util.Set;
034import javax.annotation.Nullable;
035
036/**
037 * This class provides a skeletal implementation of {@link Network}. It is recommended to extend
038 * this class rather than implement {@link Network} directly.
039 *
040 * @author James Sexton
041 * @param <N> Node parameter type
042 * @param <E> Edge parameter type
043 * @since 20.0
044 */
045@Beta
046public abstract class AbstractNetwork<N, E> implements Network<N, E> {
047
048  @Override
049  public Graph<N> asGraph() {
050    return new AbstractGraph<N>() {
051      @Override
052      public Set<N> nodes() {
053        return AbstractNetwork.this.nodes();
054      }
055
056      @Override
057      public Set<EndpointPair<N>> edges() {
058        if (allowsParallelEdges()) {
059          return super.edges(); // Defer to AbstractGraph implementation.
060        }
061
062        // Optimized implementation assumes no parallel edges (1:1 edge to EndpointPair mapping).
063        return new AbstractSet<EndpointPair<N>>() {
064          @Override
065          public Iterator<EndpointPair<N>> iterator() {
066            return Iterators.transform(
067                AbstractNetwork.this.edges().iterator(),
068                new Function<E, EndpointPair<N>>() {
069                  @Override
070                  public EndpointPair<N> apply(E edge) {
071                    return incidentNodes(edge);
072                  }
073                });
074          }
075
076          @Override
077          public int size() {
078            return AbstractNetwork.this.edges().size();
079          }
080
081          // Mostly safe: We check contains(u) before calling successors(u), so we perform unsafe
082          // operations only in weird cases like checking for an EndpointPair<ArrayList> in a
083          // Network<LinkedList>.
084          @SuppressWarnings("unchecked")
085          @Override
086          public boolean contains(@Nullable Object obj) {
087            if (!(obj instanceof EndpointPair)) {
088              return false;
089            }
090            EndpointPair<?> endpointPair = (EndpointPair<?>) obj;
091            return isDirected() == endpointPair.isOrdered()
092                && nodes().contains(endpointPair.nodeU())
093                && successors((N) endpointPair.nodeU()).contains(endpointPair.nodeV());
094          }
095        };
096      }
097
098      @Override
099      public ElementOrder<N> nodeOrder() {
100        return AbstractNetwork.this.nodeOrder();
101      }
102
103      @Override
104      public boolean isDirected() {
105        return AbstractNetwork.this.isDirected();
106      }
107
108      @Override
109      public boolean allowsSelfLoops() {
110        return AbstractNetwork.this.allowsSelfLoops();
111      }
112
113      @Override
114      public Set<N> adjacentNodes(N node) {
115        return AbstractNetwork.this.adjacentNodes(node);
116      }
117
118      @Override
119      public Set<N> predecessors(N node) {
120        return AbstractNetwork.this.predecessors(node);
121      }
122
123      @Override
124      public Set<N> successors(N node) {
125        return AbstractNetwork.this.successors(node);
126      }
127
128      // DO NOT override the AbstractGraph *degree() implementations.
129    };
130  }
131
132  @Override
133  public int degree(N node) {
134    if (isDirected()) {
135      return IntMath.saturatedAdd(inEdges(node).size(), outEdges(node).size());
136    } else {
137      return IntMath.saturatedAdd(incidentEdges(node).size(), edgesConnecting(node, node).size());
138    }
139  }
140
141  @Override
142  public int inDegree(N node) {
143    return isDirected() ? inEdges(node).size() : degree(node);
144  }
145
146  @Override
147  public int outDegree(N node) {
148    return isDirected() ? outEdges(node).size() : degree(node);
149  }
150
151  @Override
152  public Set<E> adjacentEdges(E edge) {
153    EndpointPair<N> endpointPair = incidentNodes(edge); // Verifies that edge is in this network.
154    Set<E> endpointPairIncidentEdges =
155        Sets.union(incidentEdges(endpointPair.nodeU()), incidentEdges(endpointPair.nodeV()));
156    return Sets.difference(endpointPairIncidentEdges, ImmutableSet.of(edge));
157  }
158
159  @Override
160  public Optional<E> edgeConnecting(N nodeU, N nodeV) {
161    Set<E> edgesConnecting = edgesConnecting(nodeU, nodeV);
162    switch (edgesConnecting.size()) {
163      case 0:
164        return Optional.absent();
165      case 1:
166        return Optional.of(edgesConnecting.iterator().next());
167      default:
168        throw new IllegalArgumentException(String.format(MULTIPLE_EDGES_CONNECTING, nodeU, nodeV));
169    }
170  }
171
172  @Override
173  public final boolean equals(@Nullable Object obj) {
174    if (obj == this) {
175      return true;
176    }
177    if (!(obj instanceof Network)) {
178      return false;
179    }
180    Network<?, ?> other = (Network<?, ?>) obj;
181
182    return isDirected() == other.isDirected()
183        && nodes().equals(other.nodes())
184        && edgeIncidentNodesMap(this).equals(edgeIncidentNodesMap(other));
185  }
186
187  @Override
188  public final int hashCode() {
189    return edgeIncidentNodesMap(this).hashCode();
190  }
191
192  /** Returns a string representation of this network. */
193  @Override
194  public String toString() {
195    String propertiesString =
196        String.format(
197            "isDirected: %s, allowsParallelEdges: %s, allowsSelfLoops: %s",
198            isDirected(), allowsParallelEdges(), allowsSelfLoops());
199    return String.format(
200        GRAPH_STRING_FORMAT, propertiesString, nodes(), edgeIncidentNodesMap(this));
201  }
202
203  private static <N, E> Map<E, EndpointPair<N>> edgeIncidentNodesMap(final Network<N, E> network) {
204    Function<E, EndpointPair<N>> edgeToIncidentNodesFn =
205        new Function<E, EndpointPair<N>>() {
206          @Override
207          public EndpointPair<N> apply(E edge) {
208            return network.incidentNodes(edge);
209          }
210        };
211    return Maps.asMap(network.edges(), edgeToIncidentNodesFn);
212  }
213}