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

org.jgrapht.alg.TransitiveReduction Maven / Gradle / Ivy

/* ==========================================
 * JGraphT : a free Java graph-theory library
 * ==========================================
 *
 * Project Info:  http://jgrapht.sourceforge.net/
 * Project Creator:  Barak Naveh (http://sourceforge.net/users/barak_naveh)
 *
 * (C) Copyright 2003-2008, by Barak Naveh and Contributors.
 *
 * This program and the accompanying materials are dual-licensed under
 * either
 *
 * (a) the terms of the GNU Lesser General Public License version 2.1
 * as published by the Free Software Foundation, or (at your option) any
 * later version.
 *
 * or (per the licensee's choosing)
 *
 * (b) the terms of the Eclipse Public License v1.0 as published by
 * the Eclipse Foundation.
 */
/* ----------------------
 * TransitiveReduction.java
 * ----------------------
 *
 * Original Author:   Christophe Thiebaud
 * Contributor(s):
 *
 * Changes
 * -------
 * 13-August-2015: Initial revision (CT);
 *
 */
package org.jgrapht.alg;

import java.util.*;

import org.jgrapht.*;


/**
 * An implementation of Harry Hsu's transitive
 * reduction algorithm.
 *
 * 

cf. Harry * Hsu. "An algorithm for finding a minimal equivalent graph of a digraph.", * Journal of the ACM, 22(1):11-16, January 1975.

* *

This is a port from a python example by Michael Clerx, posted as an answer * to a question about * transitive reduction algorithm pseudocode on Stack Overflow

* * @author Christophe Thiebaud * @since Aug 13, 2015 */ public class TransitiveReduction { /** * Singleton instance. */ public static final TransitiveReduction INSTANCE = new TransitiveReduction(); /** * Private Constructor. */ private TransitiveReduction() { } /** * The matrix passed as input parameter will be transformed into a path * matrix. * *

This method is package visible for unit testing, but it is meant as a * private method.

* * @param matrix the original matrix to transform into a path matrix */ static void transformToPathMatrix(BitSet [] matrix) { // compute path matrix for (int i = 0; i < matrix.length; i++) { for (int j = 0; j < matrix.length; j++) { if (i == j) { continue; } if (matrix[j].get(i)) { for (int k = 0; k < matrix.length; k++) { if (!matrix[j].get(k)) { matrix[j].set(k, matrix[i].get(k)); } } } } } } /** * The path matrix passed as input parameter will be transformed into a * transitively reduced matrix. * *

This method is package visible for unit testing, but it is meant as a * private method.

* * @param pathMatrix the path matrix to reduce */ static void transitiveReduction(BitSet [] pathMatrix) { // transitively reduce for (int j = 0; j < pathMatrix.length; j++) { for (int i = 0; i < pathMatrix.length; i++) { if (pathMatrix[i].get(j)) { for (int k = 0; k < pathMatrix.length; k++) { if (pathMatrix[j].get(k)) { pathMatrix[i].set(k, false); } } } } } } /** * This method will remove all transitive edges from the graph passed as * input parameter. * *

You may want to clone the graph before, as transitive edges will be * pitilessly removed.

* * e.g. * *
     * {
     *     @code
     *     DirectedGraph<V, T> soonToBePrunedDirectedGraph;
     *
     *     TransitiveReduction.INSTANCE.reduce(soonToBePrunedDirectedGraph);
     *
     *     // pruned !
     * }
     * 
* * @param directedGraph the directed graph that will be reduced transitively */ public void reduce(final DirectedGraph directedGraph) { final List vertices = new ArrayList(directedGraph.vertexSet()); final int n = vertices.size(); BitSet [] originalMatrix = new BitSet[n]; for (int i = 0; i < originalMatrix.length; i++) { originalMatrix[i] = new BitSet(n); } // initialize matrix with zeros // 'By default, all bits in the set initially have the value false.' // cf. http://docs.oracle.com/javase/7/docs/api/java/util/BitSet.html // initialize matrix with edges final Set edges = directedGraph.edgeSet(); for (final E edge : edges) { final V v1 = directedGraph.getEdgeSource(edge); final V v2 = directedGraph.getEdgeTarget(edge); final int v_1 = vertices.indexOf(v1); final int v_2 = vertices.indexOf(v2); originalMatrix[v_1].set(v_2); } // create path matrix from original matrix final BitSet [] pathMatrix = originalMatrix; transformToPathMatrix(pathMatrix); // create reduced matrix from path matrix final BitSet [] transitivelyReducedMatrix = pathMatrix; transitiveReduction(transitivelyReducedMatrix); // remove edges from the DirectedGraph which are not in the reduced // matrix for (int i = 0; i < n; i++) { for (int j = 0; j < n; j++) { if (!transitivelyReducedMatrix[i].get(j)) { directedGraph.removeEdge( directedGraph.getEdge( vertices.get(i), vertices.get(j))); } } } } } // End TransitiveReduction.java




© 2015 - 2025 Weber Informatics LLC | Privacy Policy