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

org.jgrapht.alg.cycle.AhujaOrlinSharmaCyclicExchangeLocalAugmentation Maven / Gradle / Ivy

/*
 * (C) Copyright 2018-2021, by Christoph Grüne 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.cycle;

import org.jgrapht.*;
import org.jgrapht.graph.*;
import org.jgrapht.util.*;

import java.util.*;

/**
 * Implementation of an algorithm for the local augmentation problem for the cyclic exchange
 * neighborhood, i.e. it finds subset-disjoint negative cycles in a graph, based on Ravindra K.
 * Ahuja, James B. Orlin, Dushyant Sharma, A composite very large-scale neighborhood structure for
 * the capacitated minimum spanning tree problem, Operations Research Letters, Volume 31, Issue 3,
 * 2003, Pages 185-194, ISSN 0167-6377, https://doi.org/10.1016/S0167-6377(02)00236-5.
 * (http://www.sciencedirect.com/science/article/pii/S0167637702002365)
 *
 * A subset-disjoint cycle is a cycle such that no two vertices in the cycle are in the same subset
 * of a given partition of the whole vertex set.
 *
 * This algorithm returns the first or the best found negative subset-disjoint cycle. In the case of
 * the first found cycle, the cycle has minimum number of vertices. It may enumerate all paths up to
 * the length given by the parameter lengthBound, i.e the algorithm runs in exponential
 * time.
 *
 * This algorithm is used to detect valid cyclic exchanges in a cyclic exchange neighborhood for the
 * Capacitated Minomum Spanning Tree problem
 * {@link org.jgrapht.alg.spanning.AhujaOrlinSharmaCapacitatedMinimumSpanningTree}
 * 
 * @see org.jgrapht.alg.spanning.AhujaOrlinSharmaCapacitatedMinimumSpanningTree
 *
 * @param  the vertex type the graph
 * @param  the edge type of the graph
 *
 * @author Christoph Grüne
 * @since June 7, 2018
 */
public class AhujaOrlinSharmaCyclicExchangeLocalAugmentation
{

    /**
     * the input graph
     */
    private Graph graph;
    /**
     * the map that maps each vertex to a subset (identified by labels) of the partition
     */
    private Map labelMap;
    /**
     * bound on how long the cycle can get
     */
    private int lengthBound;
    /**
     * contains whether the best or the first improvement is returned
     */
    private boolean bestImprovement;

    /**
     * Constructs an algorithm with given inputs
     *
     * @param graph the directed graph on which to find the negative subset disjoint cycle. The
     *        vertices of the graph are labeled according to labelMap.
     * @param lengthBound the (inclusive) upper bound for the length of cycles to detect
     * @param labelMap the labelMap of the vertices encoding the subsets of vertices
     * @param bestImprovement contains whether the best or the first improvement is returned: best
     *        if true, first if false
     */
    public AhujaOrlinSharmaCyclicExchangeLocalAugmentation(
        Graph graph, int lengthBound, Map labelMap, boolean bestImprovement)
    {
        this.graph = Objects.requireNonNull(graph, "Graph cannot be null");
        if (!graph.getType().isDirected()) {
            throw new IllegalArgumentException("The graph has to be directed.");
        }
        this.lengthBound = lengthBound;
        this.labelMap = Objects.requireNonNull(labelMap, "Labels cannot be null");
        for (V vertex : graph.vertexSet()) {
            if (!labelMap.containsKey(vertex)) {
                throw new IllegalArgumentException(
                    "Every vertex has to be labeled, that is, every vertex needs an entry in labelMap.");
            }
        }
        this.bestImprovement = bestImprovement;
    }

    /**
     * Calculates a valid subset-disjoint negative cycle. If there is no such cycle, it returns an
     * empty GraphWalk instance
     *
     * @return a valid subset-disjoint negative cycle encoded as GraphWalk
     */
    public GraphWalk getLocalAugmentationCycle()
    {

        int k = 1;

        LabeledPath bestCycle =
            new LabeledPath<>(new ArrayList<>(lengthBound), Double.MAX_VALUE, new HashSet<>());

        /*
         * Store the path in map with key PathSetKey>, since only paths with the
         * same head, same tail, and the subset of labels may be in domination relation. Thus the
         * algorithm runs faster.
         */
        Map, LabeledPath> pathsLengthK = new LinkedHashMap<>();
        Map, LabeledPath> pathsLengthKplus1 = new LinkedHashMap<>();

        // initialize pathsLengthK for k = 1
        for (E e : graph.edgeSet()) {
            if (graph.getEdgeWeight(e) < 0) {
                // initialize all paths of cost < 0
                V sourceVertex = graph.getEdgeSource(e);
                V targetVertex = graph.getEdgeTarget(e);
                // catch self-loops directly
                if (sourceVertex == targetVertex) {
                    ArrayList vertices = new ArrayList<>();
                    vertices.add(sourceVertex);
                    vertices.add(targetVertex);

                    double currentEdgeWeight = graph.getEdgeWeight(e);
                    double oppositeEdgeWeight =
                        graph.getEdgeWeight(graph.getEdge(targetVertex, sourceVertex));
                    if (bestImprovement) {
                        if (bestCycle.cost > currentEdgeWeight + oppositeEdgeWeight) {
                            HashSet labelSet = new HashSet<>();
                            labelSet.add(labelMap.get(sourceVertex));
                            bestCycle = new LabeledPath<>(
                                vertices, currentEdgeWeight + oppositeEdgeWeight, labelSet);
                        }
                    } else {
                        return new GraphWalk<>(
                            graph, vertices, currentEdgeWeight + oppositeEdgeWeight);
                    }
                }
                if (!labelMap.get(sourceVertex).equals(labelMap.get(targetVertex))) {
                    ArrayList pathVertices = new ArrayList<>(lengthBound);
                    HashSet pathLabels = new HashSet<>();
                    pathVertices.add(sourceVertex);
                    pathVertices.add(targetVertex);
                    pathLabels.add(labelMap.get(sourceVertex));
                    pathLabels.add(labelMap.get(targetVertex));
                    LabeledPath path =
                        new LabeledPath<>(pathVertices, graph.getEdgeWeight(e), pathLabels);

                    // add path to set of paths of length 1
                    updatePathIndex(pathsLengthK, path);
                }
            }
        }

        while (k < lengthBound) {

            // go through all valid paths of length k
            for (LabeledPath path : pathsLengthK.values()) {

                V head = path.getHead();
                V tail = path.getTail();

                E currentEdge = graph.getEdge(tail, head);
                if (currentEdge != null) {
                    double currentCost = path.cost + graph.getEdgeWeight(currentEdge);

                    if (currentCost < bestCycle.cost) {
                        LabeledPath cycleResult = path.clone();
                        cycleResult
                            .addVertex(head, graph.getEdgeWeight(currentEdge), labelMap.get(head));

                        /*
                         * The path builds a valid negative cycle. Return the cycle if the first
                         * improvement should be returned.
                         */
                        if (!bestImprovement && currentCost < 0) {
                            return new GraphWalk<>(graph, cycleResult.vertices, cycleResult.cost);
                        }

                        bestCycle = cycleResult;
                    }
                }

                for (E e : graph.outgoingEdgesOf(tail)) {
                    V currentVertex = graph.getEdgeTarget(e);
                    // extend the path if the extension is still negative and correctly labeled
                    double edgeWeight = graph.getEdgeWeight(e);
                    int currentLabel = labelMap.get(currentVertex);
                    if (!path.labels.contains(currentLabel) && path.cost + edgeWeight < 0) {
                        LabeledPath newPath = path.clone();
                        newPath.addVertex(currentVertex, edgeWeight, currentLabel);

                        /*
                         * check if paths are dominated, i.e. if the path is definitely worse than
                         * other paths and does not have to be considered in the future
                         */
                        if (!checkDominatedPathsOfLengthKplus1(newPath, pathsLengthKplus1)) {
                            if (!checkDominatedPathsOfLengthK(newPath, pathsLengthK)) {
                                updatePathIndex(pathsLengthKplus1, newPath);
                            }
                        }
                    }
                }

            }
            // update k and the corresponding sets
            k += 1;
            pathsLengthK = pathsLengthKplus1;
            pathsLengthKplus1 = new LinkedHashMap<>();
        }

        return new GraphWalk<>(graph, bestCycle.vertices, bestCycle.cost);
    }

    /**
     * Checks whether path dominates the current minimal cost path with the same head,
     * tail and label set in the set of all paths of length k + 1. Thus, dominated paths are
     * eliminated. This is important out of efficiency reasons, otherwise many unnecessary paths may
     * be considered in further calculations.
     *
     * @param path the currently calculated path
     * @param pathsLengthKplus1 all before calculated paths of length k + 1
     *
     * @return whether path dominates the current minimal cost path with the same head,
     *         tail and label set.
     */
    private boolean checkDominatedPathsOfLengthKplus1(
        LabeledPath path, Map, LabeledPath> pathsLengthKplus1)
    {
        // simulates domination test by using the index structure
        LabeledPath pathToCheck =
            pathsLengthKplus1.get(new PathSetKey<>(path.getHead(), path.getTail(), path.labels));
        if (pathToCheck != null) {
            return pathToCheck.cost < path.cost;
        }
        return false;
    }

    /**
     * Checks whether path is dominated by some path in the previously calculated set
     * of paths of length k. This is important out of efficiency reasons, otherwise many unnecessary
     * paths may be considered in further calculations.
     *
     * @param path the currently calculated path
     * @param pathsLengthK all previously calculated paths of length k
     *
     * @return whether path is dominated by some path in pathsLengthK
     */
    private boolean checkDominatedPathsOfLengthK(
        LabeledPath path, Map, LabeledPath> pathsLengthK)
    {
        Set modifiableLabelSet = new HashSet<>(path.labels);
        for (Integer label : path.labels) {
            modifiableLabelSet.remove(label);
            // simulates domination test by using the index structure
            LabeledPath pathToCheck = pathsLengthK
                .get(new PathSetKey<>(path.getHead(), path.getTail(), modifiableLabelSet));
            if (pathToCheck != null) {
                if (pathToCheck.cost < path.cost) {
                    return true;
                }
            }
            modifiableLabelSet.add(label);
        }
        return false;
    }

    /**
     * Adds a path and removes the path, which has the same tail, head and label set, to the data
     * structure paths, which contains all paths indexed by their head, tail and label
     * set.
     *
     * @param paths the map of paths, which are indexed by head, tail and label set, to add the path
     *        to
     * @param path the path to add
     */
    private void updatePathIndex(Map, LabeledPath> paths, LabeledPath path)
    {
        PathSetKey currentKey = new PathSetKey<>(path.getHead(), path.getTail(), path.labels);
        paths.put(currentKey, path);
    }

    /**
     * Implementation of a labeled path. It is used in
     * AhujaOrlinSharmaCyclicExchangeLocalAugmentation to efficiently maintain the paths in the
     * calculation.
     *
     * @param  the vertex type
     *
     * @author Christoph Grüne
     * @since June 7, 2018
     */
    private class LabeledPath
        implements
        Cloneable
    {

        /**
         * the vertices in the path
         */
        public ArrayList vertices;
        /**
         * the labels the path contains
         */
        public HashSet labels;
        /**
         * the cost of the path
         */
        public double cost;

        /**
         * Constructs a LabeledPath with the given inputs
         *
         * @param vertices the vertices of the path in order of the path
         * @param cost the cost of the edges connecting the vertices
         * @param labels the mapping of the vertices to labels (subsets)
         */
        public LabeledPath(ArrayList vertices, double cost, HashSet labels)
        {
            this.vertices = vertices;
            this.cost = cost;
            this.labels = labels;
        }

        /**
         * Adds a vertex to the path
         *
         * @param v the vertex
         * @param edgeCost the cost of the edge connecting the last vertex of the path and the new
         *        vertex
         * @param label the label of the new vertex
         */
        public void addVertex(V v, double edgeCost, int label)
        {
            this.vertices.add(v);
            this.cost += edgeCost;
            this.labels.add(label);
        }

        /**
         * Returns the start vertex of the path
         *
         * @return the start vertex of the path
         */
        public V getHead()
        {
            return vertices.get(0);
        }

        /**
         * Returns the end vertex of the path
         *
         * @return the end vertex of the path
         */
        public V getTail()
        {
            return vertices.get(vertices.size() - 1);
        }

        /**
         * Returns whether the path is empty, i.e. has no vertices
         *
         * @return whether the path is empty
         */
        public boolean isEmpty()
        {
            return vertices.isEmpty();
        }

        /**
         * Returns a shallow copy of this labeled path instance. Vertices are not cloned.
         *
         * @return a shallow copy of this path.
         *
         * @throws RuntimeException in case the clone is not supported
         *
         * @see java.lang.Object#clone()
         */
        public LabeledPath clone()
        {
            try {
                LabeledPath newLabeledPath = TypeUtil.uncheckedCast(super.clone());
                newLabeledPath.vertices = TypeUtil.uncheckedCast(this.vertices.clone());
                newLabeledPath.labels = TypeUtil.uncheckedCast(this.labels.clone());
                newLabeledPath.cost = this.cost;

                return newLabeledPath;
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
                throw new RuntimeException();
            }
        }
    }

    /**
     * Implementation of a key for the path maps. It is used in
     * AhujaOrlinSharmaCyclicExchangeLocalAugmentation to efficiently maintain the path sets in the
     * calculation.
     *
     * @param  the vertex type
     *
     * @author Christoph Grüne
     * @since June 7, 2018
     */
    private class PathSetKey
    {
        /**
         * the head of the paths indexed by this key
         */
        private V head;
        /**
         * the tail of the paths indexed by this key
         */
        private V tail;
        /**
         * the label set of the paths indexed by this key
         */
        private Set labels;

        /**
         * Constructs a new PathSetKey object
         *
         * @param head the head of the paths indexed by this key
         * @param tail the tail of the paths indexed by this key
         * @param labels the label set of the paths indexed by this key
         */
        private PathSetKey(V head, V tail, Set labels)
        {
            this.head = head;
            this.tail = tail;
            this.labels = labels;
        }

        @Override
        public int hashCode()
        {
            return Objects.hash(this.head, this.tail, this.labels);
        }

        @Override
        public boolean equals(Object o)
        {
            if (this == o)
                return true;
            else if (!(o instanceof PathSetKey))
                return false;

            @SuppressWarnings("unchecked") PathSetKey other = (PathSetKey) o;
            return Objects.equals(head, other.head) && Objects.equals(tail, other.tail)
                && Objects.equals(labels, other.labels);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy