找到在无向图上形成简单循环的所有路径

时间:2013-01-03 19:53:33

标签: algorithm graph cycle graph-algorithm

我在编写一个算法时遇到一些麻烦,该算法返回在无向图上形成简单循环的所有路径。

我首先考虑从顶点A开始的所有循环,对于下面的图表

A,B,E,G,F
A,B,E,D,F
A,B,C,D,F
A,B,C,D,E,G,F

其他周期将是

B,C,D,E
F,D,E,G

但是可以找到这些,例如,再次调用相同的算法,但分别从B和D开始。

图表如下所示 -

enter image description here

我目前的方法是通过访问A的所有邻居,然后访问邻居的邻居等来建立A的所有可能路径,同时遵循以下规则:

  • 每次存在多个邻居时,会找到一个分叉,并创建并探索来自A的新路径。

  • 如果任何创建的路径访问原始顶点,则该路径是一个循环。

  • 如果任何创建的路径访问相同的顶点两次(与A不同),则丢弃该路径。

  • 继续,直到探索完所有可能的路径。

我目前在尝试避免多次发现同一周期时遇到问题,我试图通过查看新邻居是否已经是另一条现有路径的一部分来解决这个问题,以便两条路径合并(如果是独立的)建立一个循环。

我的问题是:我是否遵循正确/更好/更简单的逻辑来解决这个问题。?

感谢您的评论

2 个答案:

答案 0 :(得分:4)

根据@eminsenay对other question的回答,我使用了Frank Meyer开发的elementaryCycles库,来自web_at_normalisiert_dot_de,它实现了Johnson的算法。

但是,由于这个库是针对有向图的,我添加了一些例程:

  • 从JGraphT无向图构建邻接矩阵(Meyer的lib需要)
  • 过滤结果以避免长度为2的循环
  • 删除重复循环,因为Meyer的lib用于有向图,每个无向循环是两个有向循环(每个方向一个)。

代码是

package test;

import java.util.*;

import org.jgraph.graph.DefaultEdge;
import org.jgrapht.UndirectedGraph;
import org.jgrapht.graph.SimpleGraph;


public class GraphHandling<V> {

private UndirectedGraph<V,DefaultEdge>     graph;
private List<V>                         vertexList;
private boolean                         adjMatrix[][];

public GraphHandling() {
    this.graph             = new SimpleGraph<V, DefaultEdge>(DefaultEdge.class);
    this.vertexList     = new ArrayList<V>();
}

public void addVertex(V vertex) {
    this.graph.addVertex(vertex);
    this.vertexList.add(vertex);
}

public void addEdge(V vertex1, V vertex2) {
    this.graph.addEdge(vertex1, vertex2);
}

public UndirectedGraph<V, DefaultEdge> getGraph() {
    return graph;
}

public List<List<V>>     getAllCycles() {
    this.buildAdjancyMatrix();

    @SuppressWarnings("unchecked")
    V[] vertexArray                 = (V[]) this.vertexList.toArray();
    ElementaryCyclesSearch     ecs     = new ElementaryCyclesSearch(this.adjMatrix, vertexArray);

    @SuppressWarnings("unchecked")
    List<List<V>>             cycles0    = ecs.getElementaryCycles();

    // remove cycles of size 2
    Iterator<List<V>>         listIt    = cycles0.iterator();
    while(listIt.hasNext()) {
        List<V> cycle = listIt.next();

        if(cycle.size() == 2) {
            listIt.remove();
        }
    }

    // remove repeated cycles (two cycles are repeated if they have the same vertex (no matter the order)
    List<List<V>> cycles1             = removeRepeatedLists(cycles0);

    for(List<V> cycle : cycles1) {
        System.out.println(cycle);    
    }


    return cycles1;
}

private void buildAdjancyMatrix() {
    Set<DefaultEdge>     edges        = this.graph.edgeSet();
    Integer             nVertex     = this.vertexList.size();
    this.adjMatrix                     = new boolean[nVertex][nVertex];

    for(DefaultEdge edge : edges) {
        V v1     = this.graph.getEdgeSource(edge);
        V v2     = this.graph.getEdgeTarget(edge);

        int     i = this.vertexList.indexOf(v1);
        int     j = this.vertexList.indexOf(v2);

        this.adjMatrix[i][j]     = true;
        this.adjMatrix[j][i]     = true;
    }
}
/* Here repeated lists are those with the same elements, no matter the order, 
 * and it is assumed that there are no repeated elements on any of the lists*/
private List<List<V>> removeRepeatedLists(List<List<V>> listOfLists) {
    List<List<V>> inputListOfLists         = new ArrayList<List<V>>(listOfLists);
    List<List<V>> outputListOfLists     = new ArrayList<List<V>>();

    while(!inputListOfLists.isEmpty()) {
        // get the first element
        List<V> thisList     = inputListOfLists.get(0);
        // remove it
        inputListOfLists.remove(0);
        outputListOfLists.add(thisList);
        // look for duplicates
        Integer             nEl     = thisList.size();
        Iterator<List<V>>     listIt    = inputListOfLists.iterator();
        while(listIt.hasNext()) {
            List<V>     remainingList     = listIt.next();

            if(remainingList.size() == nEl) {
                if(remainingList.containsAll(thisList)) {
                    listIt.remove();    
                }
            }
        }

    }

    return outputListOfLists;
}

}

答案 1 :(得分:3)

我之所以在你想要找到无弦周期的基础上回答这个问题,但可以修改它以找到带和弦的周期。

这个问题减少到找到两个顶点s和t之间的所有(包含)最小路径。

  • 对于所有三胞胎,(v,s,t):
    • v,s,t形成一个三角形,在这种情况下,输出它并继续下一个三元组。
    • 否则,删除除s和t以外的v及其邻居,并枚举所有s-t路径。

查找所有s-t路径可以通过动态编程来完成。