拓扑排序


对一个 有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v, 若<u,v> ∈E(G),则u在线性序列中出现在v之前

一个有向无环图的拓扑序列不是唯一的:

进行拓扑排序的算法并不复杂:

1)在有向图中选一个没有前驱(入度为0)的顶点且输出之
2)从图中删除该顶点及它发出的弧(这样就得到了别的入度为0的顶点)。

重复上述2步,直到输出全部顶点。

从描述上可以看出,我们需要记录每个顶点的入度,实现如下:由于没有记录入度这一信息,先要求出一个入度数组,来表示每个顶点的入度,这个入度数组还要动态更新,当一个顶点被删除后,它指向的顶点的入度都要减1.

在邻接表实现的有向图中,拓扑序列生成过程如下:


public Iterator Topo(DirectedGraph g){//返回拓扑排序的队列迭代器
		LinkedQueue queue = new LinkedQueue();
		int[] InDegree = new int[g.size()];//存放每个顶点的入度
		for(int i = 0;i < g.size();i++)//求出每个顶点的入度
		{
			Edge temp = VNodes[i].getFirst();
			while(temp != null)
			{
				int index = temp.getEnd();
				InDegree[index]++;
				temp = temp.getNext();
			}
		}
		
		for(int i = 0;i < g.size();i++)		//要进行n次循环找出拓扑序列
		{
			for(int j = 0;j < InDegree.length;j++)
				if(InDegree[j] == 0 && VNodes[j].getVisited() == false)
				{
					queue.enqueue(VNodes[j]);
					VNodes[j].setVisited(true);
					Edge temp = VNodes[j].getFirst();
					while(temp != null)
					{
						int index = temp.getEnd();
						InDegree[index]--;
						temp = temp.getNext();
					}
				}
		}
		
		return queue.iterator();
	}


测试一下,构造下面的有向图:

public static void main(String[] args) {
		// TODO Auto-generated method stub
		
		DirectedGraph g = new DirectedGraph();
		for(int i = 1;i < 7;i++)
			g.addVNode("V" + i);
		
		g.addEdge(0, 1);
		g.addEdge(0, 2);
		g.addEdge(0, 3);
		g.addEdge(2, 4);
		g.addEdge(2, 1);
		g.addEdge(3, 4);
		g.addEdge(5, 3);
		g.addEdge(5, 4);
		
		System.out.println("输出拓扑序列:\n");
		Iterator it = g.Topo(g);
		while(it.hasNext())
		{
			VNode node = (VNode) it.next();
			System.out.print(node.getVNode() + "  ");
		}
}



结果如下:

输出拓扑序列:

V1  V3  V6  V2  V4  V5


转载于:https://www.cnblogs.com/jmzz/archive/2011/05/25/2057270.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值