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

org.jgrapht.alg.densesubgraph.GoldbergMaximumDensitySubgraphAlgorithm Maven / Gradle / Ivy

The newest version!
/*
 * (C) Copyright 2018-2023, by Andre Immig and Contributors.
 *
 * JGraphT : a free Java graph-theory library
 *
 * See the CONTRIBUTORS.md file distributed with this work for additional
 * information regarding copyright ownership.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License 2.0 which is available at
 * http://www.eclipse.org/legal/epl-2.0, or the
 * GNU Lesser General Public License v2.1 or later
 * which is available at
 * http://www.gnu.org/licenses/old-licenses/lgpl-2.1-standalone.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR LGPL-2.1-or-later
 */
package org.jgrapht.alg.densesubgraph;

import org.jgrapht.*;
import org.jgrapht.alg.flow.*;
import org.jgrapht.alg.interfaces.*;
import org.jgrapht.graph.*;

import java.util.function.*;

/**
 * This class computes a maximum density subgraph based on the algorithm described by Andrew
 * Vladislav Goldberg in 
 * Finding Maximum Density Subgraphs, 1984, University of Berkley. 
* The basic concept is to construct a network that can be used to compute the maximum density * subgraph using a binary search approach. See {@link GoldbergMaximumDensitySubgraphAlgorithmBase} * for further details *

* This variant of the algorithm assumes the density of a positive real-weighted graph G=(V,E) to be * defined as \[\frac{\sum\limits_{e \in E} w(e)}{\left|{V}\right|}\] and sets the weights of the * network from {@link GoldbergMaximumDensitySubgraphAlgorithmBase} as proposed in the above paper. * For this case the weights of the network must be chosen to be: \[c_{ij}=w(ij)\,\forall \{i,j\}\in * E\] \[c_{it}=m+2g-d_i\,\forall i \in V\] \[c_{si}=m\,\forall i \in V\] where $m=\left|{E}\right|$ * and $d_i$ is the degree of vertex $i$.
* All the math to prove the correctness of these weights is the same as in * {@link GoldbergMaximumDensitySubgraphAlgorithmBase}.
*

* Because the density is per definition guaranteed to be rational, the distance of 2 possible * solutions for the maximum density can't be smaller than $\frac{1}{W(W-1)}$. This means shrinking * the binary search interval to this size, the correct solution is found. The runtime can in this * case be given by $O(M(n,n+m)\log{W})$, where $M(n,m)$ is the runtime of the internally used * {@link MinimumSTCutAlgorithm} and $W$ is the sum of all weights from $G$. *

* * @param Type of vertices * @param Type of edges * * @author Andre Immig */ public class GoldbergMaximumDensitySubgraphAlgorithm extends GoldbergMaximumDensitySubgraphAlgorithmBase { /** * Constructor * * @param algFactory factory to construct the algorithm to use * @param graph input for computation * @param s additional source vertex * @param t additional target vertex * @param epsilon to use for internal computation */ public GoldbergMaximumDensitySubgraphAlgorithm( Graph graph, V s, V t, double epsilon, Function, MinimumSTCutAlgorithm> algFactory) { super(graph, s, t, false, epsilon, algFactory); } /** * Convenience constructor that uses PushRelabel as default MinimumSTCutAlgorithm * * @param graph input for computation * @param s additional source vertex * @param t additional target vertex * @param epsilon to use for internal computation */ public GoldbergMaximumDensitySubgraphAlgorithm(Graph graph, V s, V t, double epsilon) { this(graph, s, t, epsilon, PushRelabelMFImpl::new); } /** * Getter for network weights of edges su for u in V * * @param v of V * @return weight of the edge */ protected double getEdgeWeightFromSourceToVertex(V v) { return this.graph.edgeSet().size(); } /** * Getter for network weights of edges ut for u in V * * @param v of V * @return weight of the edge */ protected double getEdgeWeightFromVertexToSink(V v) { return this.graph.edgeSet().size() + 2 * guess - this.graph.outgoingEdgesOf(v).stream().mapToDouble(this.graph::getEdgeWeight).sum(); } @Override protected double computeDensityNumerator(Graph g) { return g.edgeSet().stream().mapToDouble(g::getEdgeWeight).sum(); } @Override protected double computeDensityDenominator(Graph g) { return g.vertexSet().size(); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy