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.graph.GraphConstants.NOT_AVAILABLE_ON_UNDIRECTED; 021 022import com.google.common.annotations.Beta; 023import com.google.common.base.Objects; 024import com.google.common.collect.Iterators; 025import com.google.common.collect.UnmodifiableIterator; 026import com.google.errorprone.annotations.Immutable; 027import org.jspecify.annotations.Nullable; 028 029/** 030 * An immutable pair representing the two endpoints of an edge in a graph. The {@link EndpointPair} 031 * of a directed edge is an ordered pair of nodes ({@link #source()} and {@link #target()}). The 032 * {@link EndpointPair} of an undirected edge is an unordered pair of nodes ({@link #nodeU()} and 033 * {@link #nodeV()}). 034 * 035 * <p>The edge is a self-loop if, and only if, the two endpoints are equal. 036 * 037 * @author James Sexton 038 * @since 20.0 039 */ 040@Beta 041@Immutable(containerOf = {"N"}) 042public abstract class EndpointPair<N> implements Iterable<N> { 043 private final N nodeU; 044 private final N nodeV; 045 046 private EndpointPair(N nodeU, N nodeV) { 047 this.nodeU = checkNotNull(nodeU); 048 this.nodeV = checkNotNull(nodeV); 049 } 050 051 /** Returns an {@link EndpointPair} representing the endpoints of a directed edge. */ 052 public static <N> EndpointPair<N> ordered(N source, N target) { 053 return new Ordered<>(source, target); 054 } 055 056 /** Returns an {@link EndpointPair} representing the endpoints of an undirected edge. */ 057 public static <N> EndpointPair<N> unordered(N nodeU, N nodeV) { 058 // Swap nodes on purpose to prevent callers from relying on the "ordering" of an unordered pair. 059 return new Unordered<>(nodeV, nodeU); 060 } 061 062 /** Returns an {@link EndpointPair} representing the endpoints of an edge in {@code graph}. */ 063 static <N> EndpointPair<N> of(Graph<?> graph, N nodeU, N nodeV) { 064 return graph.isDirected() ? ordered(nodeU, nodeV) : unordered(nodeU, nodeV); 065 } 066 067 /** Returns an {@link EndpointPair} representing the endpoints of an edge in {@code network}. */ 068 static <N> EndpointPair<N> of(Network<?, ?> network, N nodeU, N nodeV) { 069 return network.isDirected() ? ordered(nodeU, nodeV) : unordered(nodeU, nodeV); 070 } 071 072 /** 073 * If this {@link EndpointPair} {@link #isOrdered()}, returns the node which is the source. 074 * 075 * @throws UnsupportedOperationException if this {@link EndpointPair} is not ordered 076 */ 077 public abstract N source(); 078 079 /** 080 * If this {@link EndpointPair} {@link #isOrdered()}, returns the node which is the target. 081 * 082 * @throws UnsupportedOperationException if this {@link EndpointPair} is not ordered 083 */ 084 public abstract N target(); 085 086 /** 087 * If this {@link EndpointPair} {@link #isOrdered()} returns the {@link #source()}; otherwise, 088 * returns an arbitrary (but consistent) endpoint of the origin edge. 089 */ 090 public final N nodeU() { 091 return nodeU; 092 } 093 094 /** 095 * Returns the node {@link #adjacentNode(Object) adjacent} to {@link #nodeU()} along the origin 096 * edge. If this {@link EndpointPair} {@link #isOrdered()}, this is equal to {@link #target()}. 097 */ 098 public final N nodeV() { 099 return nodeV; 100 } 101 102 /** 103 * Returns the node that is adjacent to {@code node} along the origin edge. 104 * 105 * @throws IllegalArgumentException if this {@link EndpointPair} does not contain {@code node} 106 * @since 20.0 (but the argument type was changed from {@code Object} to {@code N} in 31.0) 107 */ 108 public final N adjacentNode(N node) { 109 if (node.equals(nodeU)) { 110 return nodeV; 111 } else if (node.equals(nodeV)) { 112 return nodeU; 113 } else { 114 throw new IllegalArgumentException("EndpointPair " + this + " does not contain node " + node); 115 } 116 } 117 118 /** 119 * Returns {@code true} if this {@link EndpointPair} is an ordered pair (i.e. represents the 120 * endpoints of a directed edge). 121 */ 122 public abstract boolean isOrdered(); 123 124 /** Iterates in the order {@link #nodeU()}, {@link #nodeV()}. */ 125 @Override 126 public final UnmodifiableIterator<N> iterator() { 127 return Iterators.forArray(nodeU, nodeV); 128 } 129 130 /** 131 * Two ordered {@link EndpointPair}s are equal if their {@link #source()} and {@link #target()} 132 * are equal. Two unordered {@link EndpointPair}s are equal if they contain the same nodes. An 133 * ordered {@link EndpointPair} is never equal to an unordered {@link EndpointPair}. 134 */ 135 @Override 136 public abstract boolean equals(@Nullable Object obj); 137 138 /** 139 * The hashcode of an ordered {@link EndpointPair} is equal to {@code Objects.hashCode(source(), 140 * target())}. The hashcode of an unordered {@link EndpointPair} is equal to {@code 141 * nodeU().hashCode() + nodeV().hashCode()}. 142 */ 143 @Override 144 public abstract int hashCode(); 145 146 private static final class Ordered<N> extends EndpointPair<N> { 147 private Ordered(N source, N target) { 148 super(source, target); 149 } 150 151 @Override 152 public N source() { 153 return nodeU(); 154 } 155 156 @Override 157 public N target() { 158 return nodeV(); 159 } 160 161 @Override 162 public boolean isOrdered() { 163 return true; 164 } 165 166 @Override 167 public boolean equals(@Nullable Object obj) { 168 if (obj == this) { 169 return true; 170 } 171 if (!(obj instanceof EndpointPair)) { 172 return false; 173 } 174 175 EndpointPair<?> other = (EndpointPair<?>) obj; 176 if (isOrdered() != other.isOrdered()) { 177 return false; 178 } 179 180 return source().equals(other.source()) && target().equals(other.target()); 181 } 182 183 @Override 184 public int hashCode() { 185 return Objects.hashCode(source(), target()); 186 } 187 188 @Override 189 public String toString() { 190 return "<" + source() + " -> " + target() + ">"; 191 } 192 } 193 194 private static final class Unordered<N> extends EndpointPair<N> { 195 private Unordered(N nodeU, N nodeV) { 196 super(nodeU, nodeV); 197 } 198 199 @Override 200 public N source() { 201 throw new UnsupportedOperationException(NOT_AVAILABLE_ON_UNDIRECTED); 202 } 203 204 @Override 205 public N target() { 206 throw new UnsupportedOperationException(NOT_AVAILABLE_ON_UNDIRECTED); 207 } 208 209 @Override 210 public boolean isOrdered() { 211 return false; 212 } 213 214 @Override 215 public boolean equals(@Nullable Object obj) { 216 if (obj == this) { 217 return true; 218 } 219 if (!(obj instanceof EndpointPair)) { 220 return false; 221 } 222 223 EndpointPair<?> other = (EndpointPair<?>) obj; 224 if (isOrdered() != other.isOrdered()) { 225 return false; 226 } 227 228 // Equivalent to the following simple implementation: 229 // boolean condition1 = nodeU().equals(other.nodeU()) && nodeV().equals(other.nodeV()); 230 // boolean condition2 = nodeU().equals(other.nodeV()) && nodeV().equals(other.nodeU()); 231 // return condition1 || condition2; 232 if (nodeU().equals(other.nodeU())) { // check condition1 233 // Here's the tricky bit. We don't have to explicitly check for condition2 in this case. 234 // Why? The second half of condition2 requires that nodeV equals other.nodeU. 235 // We already know that nodeU equals other.nodeU. Combined with the earlier statement, 236 // and the transitive property of equality, this implies that nodeU equals nodeV. 237 // If nodeU equals nodeV, condition1 == condition2, so checking condition1 is sufficient. 238 return nodeV().equals(other.nodeV()); 239 } 240 return nodeU().equals(other.nodeV()) && nodeV().equals(other.nodeU()); // check condition2 241 } 242 243 @Override 244 public int hashCode() { 245 return nodeU().hashCode() + nodeV().hashCode(); 246 } 247 248 @Override 249 public String toString() { 250 return "[" + nodeU() + ", " + nodeV() + "]"; 251 } 252 } 253}