1

我正在做一个由 JUNG2 绘制的网络拓扑项目,现在发现 JUNG2 无法在不同的顶点上绘制相同的边(名称)。报告如下:

Exception in thread "main" java.lang.IllegalArgumentException: edge GigabitEthernet0/3<--->GigabitEthernet0/0/0/0 already exists in this graph with endpoints <HKBR1, HKBR3> and cannot be added with endpoints <HKBR2, HKBR4>
at edu.uci.ics.jung.graph.AbstractGraph.getValidatedEndpoints(AbstractGraph.java:93)
at edu.uci.ics.jung.graph.SparseMultigraph.addEdge(SparseMultigraph.java:123)
at edu.uci.ics.jung.graph.AbstractGraph.addEdge(AbstractGraph.java:60)
at pkg.DrawnTopology.DrawnTopology(DrawnTopology.java:45)
at pkg.ReadLine.main(ReadLine.java:85)

我已将所有边和顶点存储到一个二维数组中。二维数组数据从不同的文件中学习,所以数据是动态的。由于网络拓扑可能具有相同的边缘名称,那么如何解决呢?

4

2 回答 2

3

这个问题显然缺乏细节,但从描述和错误消息中,人们可能会猜到这里出了什么问题:

您的图表可能被定义为Graph<V, String>.

它可能是 的子类Graph,顶点类型无关紧要。关键是边缘类型是String)

您正在尝试向图中添加边,如下所示:

g.addEdge("edge0", v0, v1);
g.addEdge("edge0", v2, v3);

这将导致错误,因为边缘"edge0"存在两次。(这是有道理的。否则:当您向图形询问 的端点时会发生什么?它"edge0"应该返回v0,v1还是v2,v3?这只是模棱两可)。

(请注意,这不仅适用于边类型为 的情况。只要两条边相互连接,任何String边类型都会发生这种情况)。equal

这里一个简单的解决方案是引入一个Edge包装字符串的专用类:

class Edge
{
    private final String name;

    Edge(String name)
    {
        this.name = name;
    }

    @Override
    public String toString()
    {
        return name;
    }
}

此类没有重写equals其方法。因此,即使字符串相等,这种类型的两个对象也不相等:

Edge e0 = new Edge("edge0");
Edge e1 = new Edge("edge0");
System.out.println(e0.equals(e1)); // prints "false"

然后边缘标签可以简单地是toString这些边缘的表示,它们返回原始字符串。

一个例子:

import javax.swing.JFrame;

import org.apache.commons.collections15.Transformer;

import edu.uci.ics.jung.algorithms.layout.FRLayout;
import edu.uci.ics.jung.graph.DirectedSparseGraph;
import edu.uci.ics.jung.graph.Graph;
import edu.uci.ics.jung.visualization.VisualizationViewer;

class Edge
{
    private final String name;

    Edge(String name)
    {
        this.name = name;
    }

    @Override
    public String toString()
    {
        return name;
    }
}

public class JungDuplicateEdgesTest
{
    public static void main(String[] args)
    {
        JFrame jf = new JFrame();
        final Graph<String, Edge> g = getGraph();
        VisualizationViewer<String, Edge> vv = 
            new VisualizationViewer<String, Edge>(
                new FRLayout<String, Edge>(g));

        class EdgeLabelTransformer implements Transformer<Edge, String>
        {
            @Override
            public String transform(Edge edge)
            {
                return edge.toString();
            }
        }
        vv.getRenderContext().setEdgeLabelTransformer(
            new EdgeLabelTransformer());

        jf.getContentPane().add(vv);
        jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        jf.pack();
        jf.setVisible(true);
    }

    public static Graph<String, Edge> getGraph()
    {
        Graph<String, Edge> g = new DirectedSparseGraph<String, Edge>();
        g.addVertex("v0");
        g.addVertex("v1");
        g.addEdge(new Edge("e0"), "v0", "v1");
        g.addEdge(new Edge("e0"), "v1", "v0");
        return g;
    }
}

附注:对于某些应用程序,将边的顶点存储在Edge类中也是有意义的。然后,您可以实现equalsandhashCode方法,当它们具有相同的顶点时,您可以将它们视为相等的。

于 2015-05-28T10:55:52.150 回答
0

从技术上讲,边缘名称必须是唯一的,但您可以操纵图表中显示的内容。为了获得边的唯一名称,我使用以下模式来命名边:fromVertex_edgeName_toVertex。

然后你需要一个EdgeLabelTransformer

private class EdgeLabelTransformer<V> implements Transformer<V, String> {

    @Override
    public String transform(V v) {
        return v.toString().substring(v.toString().indexOf("_") + 1, v.toString().lastIndexOf("_"));
    }
}

这个转换器实际上告诉了VisualViewer要显示的内容。因此,在我的情况下,它只显示中间部分,这导致您可以在图表中拥有多个具有相同显示名称的边。

像这样添加变压器:

visualViewer.getRenderContext().setEdgeLabelTransformer(new EdgeLabelTransformer<String>());
于 2015-05-28T06:35:00.737 回答