All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.googlecode.blaisemath.graph.SparseGraph Maven / Gradle / Ivy

The newest version!
/*
 * SparseGraph.java
 * Created May 21, 2010
 */

package com.googlecode.blaisemath.graph;

/*
 * #%L
 * BlaiseGraphTheory
 * --
 * Copyright (C) 2009 - 2016 Elisha Peterson
 * --
 * 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.
 * #L%
 */

import com.google.common.collect.HashBasedTable;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.Lists;
import com.google.common.collect.Multimap;
import com.google.common.collect.SetMultimap;
import com.google.common.collect.Table;
import com.googlecode.blaisemath.util.Edge;
import com.googlecode.blaisemath.util.Edge.UndirectedEdge;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.Map.Entry;
import java.util.Set;
import javax.annotation.concurrent.Immutable;

/**
 * 

* General graph structure that maintains cached information to speed graph computations. * Maintains a set of all edges in the graph, a map associating each vertex with its adjacent edges, * and information about connected components. The graph cannot be changed once it is created. *

* * @param the type of the nodes * * @author Elisha Peterson */ @Immutable public final class SparseGraph extends GraphSupport { /** Edges in graph (replicated for speed) */ private final Set> edges = new LinkedHashSet>(); /** * Edges in graph, by vertex. Each vertex contains all edges adjacent to the * vertex, regardless of direction. */ private final SetMultimap> edgeIndex = HashMultimap.create(); /** * The adjacencies in components of the graph. If directed, rows are sources * and columns are destinations. Each pair of vertices may have multiple edges. * If undirected, the set of edges is the same for both directions. */ private final Table>> edgeTable = HashBasedTable.create(); /** Information about the graph's components (replicated for speed) */ private GraphComponents components; /** * Helper constructor for factory methods * @param directed whether graph is directed * @param nodes nodes in the graph */ private SparseGraph(boolean directed, Iterable nodes) { super(directed, nodes); } /** * Construct graph with specific nodes and edges. * @param graph node type * @param directed whether graph is directed * @param nodes nodes in the graph * @param edges edges in the graph, as ordered node pairs; each must have a 0 element and a 1 element * @return created graph */ public static SparseGraph createFromEdges(boolean directed, Iterable nodes, Iterable> edges) { SparseGraph res = new SparseGraph(directed, Lists.newArrayList(nodes)); for (Edge e : edges) { res.addEdge(e.getNode1(), e.getNode2()); } res.components = new GraphComponents(res, GraphUtils.components(res.edgeTable)); return res; } // // FACTORY METHODS // /** * Construct graph with specific nodes and edges * @param graph node type * @param directed whether graph is directed * @param nodes nodes in the graph * @param edges edges in the graph, as ordered node pairs; each must have a 0 element and a 1 element * @return graph */ public static SparseGraph createFromArrayEdges(boolean directed, V[] nodes, Iterable edges) { SparseGraph res = new SparseGraph(directed, Arrays.asList(nodes)); for (V[] e : edges) { res.addEdge(e[0], e[1]); } res.components = new GraphComponents(res, GraphUtils.components(res.edgeTable)); return res; } /** * Construct graph with specific nodes and edges * @param graph node type * @param directed whether graph is directed * @param nodes nodes in the graph * @param edges edges in the graph, as ordered node pairs; each must have a 0 element and a 1 element * @return graph */ public static SparseGraph createFromArrayEdges(boolean directed, Iterable nodes, Iterable edges) { SparseGraph res = new SparseGraph(directed, nodes); for (V[] e : edges) { res.addEdge(e[0], e[1]); } res.components = new GraphComponents(res, GraphUtils.components(res.edgeTable)); return res; } /** * Construct graph with a sparse adjacency representation. * @param graph node type * @param directed if graph is directed * @param adjacencies map with adjacency info * @return graph */ public static SparseGraph createFromAdjacencies(boolean directed, Multimap adjacencies) { SparseGraph res = new SparseGraph(directed, GraphUtils.nodes(adjacencies)); for (Entry en : adjacencies.entries()) { res.addEdge(en.getKey(), en.getValue()); } res.components = new GraphComponents(res, GraphUtils.components(res.edgeTable)); return res; } // /** Invoke from initializer only */ private void addEdge(V x, V y) { Edge edge = directed ? addDirectedEdge(x, y) : addUndirectedEdge(x, y); edges.add(edge); edgeIndex.put(x, edge); edgeIndex.put(y, edge); } /** Invoke from initializer only */ private Edge addDirectedEdge(V x, V y) { if (!edgeTable.contains(x, y)) { edgeTable.put(x, y, new HashSet>()); } Edge edge = new Edge(x, y); edgeTable.get(x, y).add(edge); return edge; } /** Invoke from initializer only */ private Edge addUndirectedEdge(V x, V y) { if (!edgeTable.contains(x, y)) { edgeTable.put(x, y, new HashSet>()); } if (!edgeTable.contains(y, x)) { edgeTable.put(y, x, new HashSet>()); } UndirectedEdge edge = new UndirectedEdge(x, y); edgeTable.get(x, y).add(edge); edgeTable.get(y, x).add(edge); return edge; } // @Override public String toString() { return String.format("SparseGraph[%s,%d nodes,%d edges]", directed?"directed":"undirected", nodeCount(), edgeCount()); } @Override public boolean adjacent(V x, V y) { if (edgeTable.contains(x, y) && !edgeTable.get(x, y).isEmpty()) { return true; } if (directed && edgeTable.contains(y, x) && !edgeTable.get(y, x).isEmpty()) { return true; } return false; } public GraphComponents getComponentInfo() { return components; } @Override public Set> edges() { return Collections.unmodifiableSet(edges); } @Override public Collection> edgesAdjacentTo(V x) { return Collections.unmodifiableSet(edgeIndex.get(x)); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy