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

org.jgrapht.alg.clique.ChordalGraphMaxCliqueFinder Maven / Gradle / Ivy

The newest version!
/*
 * (C) Copyright 2018-2023, by Timofey Chudakov 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.clique;

import org.jgrapht.*;
import org.jgrapht.alg.color.*;
import org.jgrapht.alg.cycle.*;
import org.jgrapht.alg.interfaces.*;
import org.jgrapht.traverse.*;
import org.jgrapht.util.*;

import java.util.*;

/**
 * Calculates a maximum cardinality
 * clique in a chordal graph. A
 * chordal graph is a simple graph in which all
 *  cycles of four or more vertices have
 * a  chord. A chord is an edge that is
 * not part of the cycle but connects two vertices of the cycle.
 *
 * To compute the clique, this implementation relies on the {@link ChordalityInspector} to compute a
 * 
 * perfect elimination order.
 *
 * The maximum clique for a chordal graph is computed in $\mathcal{O}(|V| + |E|)$ time.
 *
 * All the methods in this class are invoked in a lazy fashion, meaning that computations are only
 * started once the method gets invoked.
 *
 * @param  the graph vertex type.
 * @param  the graph edge type.
 *
 * @author Timofey Chudakov
 */
public class ChordalGraphMaxCliqueFinder
    implements CliqueAlgorithm
{
    private final Graph graph;
    private final ChordalityInspector.IterationOrder iterationOrder;

    private Clique maximumClique;
    private boolean isChordal = true;

    /**
     * Creates a new ChordalGraphMaxCliqueFinder instance. The {@link ChordalityInspector} used in
     * this implementation uses the default {@link MaximumCardinalityIterator} iterator.
     *
     * @param graph graph
     */
    public ChordalGraphMaxCliqueFinder(Graph graph)
    {
        this(graph, ChordalityInspector.IterationOrder.MCS);
    }

    /**
     * Creates a new ChordalGraphMaxCliqueFinder instance. The {@link ChordalityInspector} used in
     * this implementation uses either the {@link MaximumCardinalityIterator} iterator or the
     * {@link LexBreadthFirstIterator} iterator, depending on the parameter {@code iterationOrder}.
     *
     * @param graph graph
     * @param iterationOrder constant which defines iterator to be used by the
     *        {@code ChordalityInspector} in this implementation.
     */
    public ChordalGraphMaxCliqueFinder(
        Graph graph, ChordalityInspector.IterationOrder iterationOrder)
    {
        this.graph = Objects.requireNonNull(graph);
        this.iterationOrder = Objects.requireNonNull(iterationOrder);
    }

    /**
     * Lazily computes some maximum clique of the {@code graph}.
     */
    private void lazyComputeMaximumClique()
    {
        if (maximumClique == null && isChordal) {
            ChordalGraphColoring cgc = new ChordalGraphColoring<>(graph, iterationOrder);
            VertexColoringAlgorithm.Coloring coloring = cgc.getColoring();
            List perfectEliminationOrder = cgc.getPerfectEliminationOrder();
            if (coloring == null) {
                isChordal = false; // Graph isn't chordal
                return;
            }
            // finds the vertex with the maximum cardinality predecessor list
            Map vertexInOrder = getVertexInOrder(perfectEliminationOrder);
            Map.Entry maxEntry = coloring
                    .getColors().entrySet().stream().max(Comparator.comparing(Map.Entry::getValue))
                    .orElse(null);
            if (maxEntry == null) {
                maximumClique = new CliqueImpl<>(Collections.emptySet());
            } else {
                Set cliqueSet = getPredecessors(vertexInOrder, maxEntry.getKey());
                cliqueSet.add(maxEntry.getKey());
                maximumClique = new CliqueImpl<>(cliqueSet);
            }
        }
    }

    /**
     * Returns a map containing vertices from the {@code vertexOrder} mapped to their indices in
     * {@code vertexOrder}.
     *
     * @param vertexOrder a list with vertices.
     * @return a mapping of vertices from {@code vertexOrder} to their indices in
     *         {@code vertexOrder}.
     */
    private Map getVertexInOrder(List vertexOrder)
    {
        Map vertexInOrder =
            CollectionUtil.newHashMapWithExpectedSize(vertexOrder.size());
        int i = 0;
        for (V vertex : vertexOrder) {
            vertexInOrder.put(vertex, i++);
        }
        return vertexInOrder;
    }

    /**
     * Returns the predecessors of {@code vertex} in the order defined by {@code map}. More
     * precisely, returns those of {@code vertex}, whose mapped index in {@code map} is less then
     * the index of {@code vertex}.
     *
     * @param vertexInOrder defines the mapping of vertices in {@code graph} to their indices in
     *        order.
     * @param vertex the vertex whose predecessors in order are to be returned.
     * @return the predecessors of {@code vertex} in order defines by {@code map}.
     */
    private Set getPredecessors(Map vertexInOrder, V vertex)
    {
        Set predecessors = new HashSet<>();
        Integer vertexPosition = vertexInOrder.get(vertex);
        Set edges = graph.edgesOf(vertex);
        for (E edge : edges) {
            V oppositeVertex = Graphs.getOppositeVertex(graph, edge, vertex);
            Integer destPosition = vertexInOrder.get(oppositeVertex);
            if (destPosition < vertexPosition)
                predecessors.add(oppositeVertex);
        }
        return predecessors;
    }

    /**
     * Returns a maximum cardinality
     * clique of the inspected {@code graph}. If the graph isn't chordal, returns null.
     *
     * @return a maximum clique of the {@code graph} if it is chordal, null otherwise.
     */
    public Clique getClique()
    {
        lazyComputeMaximumClique();
        return maximumClique;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy