Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4247622
  • 博文数量: 176
  • 博客积分: 10059
  • 博客等级: 上将
  • 技术积分: 4681
  • 用 户 组: 普通用户
  • 注册时间: 2006-03-24 12:27
文章分类

全部博文(176)

文章存档

2012年(1)

2011年(4)

2010年(14)

2009年(71)

2008年(103)

分类: Java

2008-04-12 16:43:23

     图(Graph)是一种复杂的非线性结构。在人工智能、工程、数学、物理、化学、生物和计算机科学等领域中,图结构有着广泛的应用。
     本章先介绍图的概念,再介绍图的存储方法及有关图的算法。

图的二元组定义

     图G由两个集合V和E组成,记为:
        G=(V,E)
  其中:
  V是顶点的有穷非空集合,
  E是V中顶点偶对(称为边)的有穷集。
     通常,也将图G的顶点集和边集分别记为V(G)和E(G)。E(G)可以是空集。若E(G)为空,则图G只有顶点而没有边。

有向图和无向图

1.有向图

     若图G中的每条边都是有方向的,则称G为有向图(Digraph)。
(1)有向边的表示
     在有向图中,一条有向边是由两个顶点组成的有序对,有序对通常用尖括号表示。有向边也称为弧(Arc),边的始点称为弧尾(Tail),终点称为弧头(Head)。
  【例】i,vj>表示一条有向边,vi是边的始点(起点),vj是边的终点。因此,i,vj>和j,vi>是两条不同的有向边。

(2)有向图的表示
  【例】下面(a)图中G1是一个有向图。图中边的方向是用从始点指向终点的箭头表示的,该图的顶点集和边集分别为:
      V(G1)={v1,v2,v3}
      E(G1)={1,v2>,2,v1>,2,v3>}
2.无向图
     若图G中的每条边都是没有方向的,则称G为无向图(Undigraph)。
(1)无向边的表示
     无向图中的边均是顶点的无序对,无序对通常用圆括号表示。
  【例】无序对(vi,vj)和(vj,vi)表示同一条边。

(2)无向图的表示
 
【例】下面(b)图中的G2和(c)图中的G3均是无向图,它们的顶点集和边集分别为:
    V(G2)={v1,v2,v3,v4}
    E(G2)={(vl,v2),(v1,v3),(v1,v4),(v2,v3),(v2,v4),(v3,v4)}
    V(G3)={v1,v2,v3,v4,v5,v6,v7}
    E(G3)={(v1,v2),(vl,v3),(v2,v4),(v2,v5),(v3,v6),(v3,v7)}
  注意:
    
在以下讨论中,不考虑顶点到其自身的边。即若(v1,v2)或l,v2>是E(G)中的一条边,则要求v1≠v2。此外,不允许一条边在图中重复出现,即只讨论简单的图。

3.图G的顶点数n和边数e的关系
(1)若G是无向图,则0≤e≤n(n-1)/2
     恰有n(n-1)/2条边的无向图称无向完全图(Undireet-ed Complete Graph)

(2)若G是有向图,则0≤e≤n(n-1)。
     恰有n(n-1)条边的有向图称为有向完全图(Directed Complete Graph)。
  注意:
     完全图具有最多的边数。任意一对顶点间均有边相连。
  【例】上面(b)图的G2就是具有4个顶点的无向完全图。


4、图的邻接矩阵表示法
   
     在图的邻接矩阵表示法中:
 ① 用邻接矩阵表示顶点间的相邻关系
 ② 用一个顺序表来存储顶点信息


5、图的基本算法的Java实现

包括遍历、最小生成树等。



package zieckey.datastructure.study.graph;

/**
 *
 * 图中的(节)点。
 *
 * @author zieckey
 *
 */

class Vertex
{
    public char        label;        // label (e.g. 'A')

    public boolean    wasVisited;

    public Vertex( char lab ) // constructor

    {
        label = lab;
        wasVisited = false;
    }
    
    
    
} // end class Vertex

package zieckey.datastructure.study.graph;

class Graph
{
    private final int    MAX_VERTS    = 20;
    private Vertex[]    vertexList;        // array of vertices

    private int[][]        adjMat;            // adjacency matrix

    private int            nVerts;            // current number of vertices

    private StackInt    theStack;
    private QueueInt    theQueue;

    /**
     * 默认构造函数 初始化邻接矩阵
     */

    public Graph( )
    {
        vertexList = new Vertex[MAX_VERTS];

        // 图的邻接矩阵adjacency matrix

        adjMat = new int[MAX_VERTS][MAX_VERTS];
        nVerts = 0;
        for ( int j = 0; j < MAX_VERTS; j++ )
        {
            // set adjacency

            for ( int k = 0; k < MAX_VERTS; k++ )
            {
                // matrix to 0

                adjMat[j][k] = 0;
            }
        }

        theStack = new StackInt( MAX_VERTS );
        theQueue = new QueueInt( MAX_VERTS );
    }

    /**
     * 添加一个节点
     *
     * @param lab
     */

    public void addVertex( char lab ) // argument is label

    {
        vertexList[nVerts++ ] = new Vertex( lab );
    }

    /**
     * 添加一条边
     *
     * @param start
     * 起始点
     * @param end
     * 终点
     */

    public void addEdge( int start, int end )
    {
        adjMat[start][end] = 1;
        adjMat[end][start] = 1;
    }

    /**
     * 添加一条边
     *
     * @param start
     * 起始点
     * @param end
     * 终点
     */

    public void addEdge( char startVertex, char endVertex )
    {
        int start = getIndexByLabel( startVertex );
        int end = getIndexByLabel( endVertex );
        if ( -1 != start && -1 != end )
        {
            adjMat[start][end] = 1;
            adjMat[end][start] = 1;
        }
    }

    /**
     * depth-first search深度优先搜索
     */

    public void dfs()
    {
        theStack.setTop( -1 );// 置空stack

        vertexList[0].wasVisited = true; // mark it

        displayVertex( 0 ); // display it

        theStack.push( 0 ); // push it


        int v = 0;
        while ( !theStack.isEmpty() )
        {
            v = getAdjUnvisitedVertex( theStack.peek() );
            if ( -1 == v )
            {
                theStack.pop();
            } else
            {
                vertexList[v].wasVisited = true;
                displayVertex( v );
                theStack.push( v );
            }
        }// end while


        // 重置标志

        for ( int j = 0; j < nVerts; j++ )
        {
            vertexList[j].wasVisited = false;
        }
    }

    /**
     * breadth-first search 广度优先搜索
     */

    public void bfs()
    {
        vertexList[0].wasVisited = true; // mark it

        theQueue.insert( 0 );

        
        while ( !theQueue.isEmpty() )
        {
            int currentVertex = theQueue.remove();
            displayVertex( currentVertex );//打印当前访问的节点

            int v = -2;
            
            //搜索与当前节点currentVertex相邻的所有节点

            while ( -1 != ( v = getAdjUnvisitedVertex( currentVertex ) ) )
            {
                theQueue.insert( v );
                vertexList[v].wasVisited = true; // mark it

            }
        }
        
        // 重置标志

        for ( int j = 0; j < nVerts; j++ )
        {
            vertexList[j].wasVisited = false;
        }
    }
    
    /**
     * 最小生成树算法。用深度优先搜索算法得到
     */

    public void buildMinimumSpanningTree()
    {
        theStack.setTop( -1 );// 置空stack

        vertexList[0].wasVisited = true; // mark it

        theStack.push( 0 ); // push it


        int v = 0;
        while ( !theStack.isEmpty() )
        {
            int currentVertex = theStack.peek();
            v = getAdjUnvisitedVertex( currentVertex );
            if ( -1 == v )
            {
                theStack.pop();
            } else
            {
                vertexList[v].wasVisited = true;
                displayVertex( currentVertex );
                displayVertex( v );
                System.out.print(" ");
                theStack.push( v );
            }
        }// end while


        // 重置标志

        for ( int j = 0; j < nVerts; j++ )
        {
            vertexList[j].wasVisited = false;
        }
    }

    /**
     * 得到节点 v 的还未被访问的一个节点
     *
     * @param v
     * @return 返回v的还未被访问的一个节点,如果没有找到,返回-1
     */

    public int getAdjUnvisitedVertex( int v )
    {
        for ( int j = 0; j < nVerts; j++ )
        {
            if ( adjMat[v][j] == 1 && vertexList[j].wasVisited == false )
            {
                return j; // return first such vertex

            }
        }
        return -1; // no such vertices

    } // end getAdjUnvisitedVert()


    /**
     * 显示一个节点
     *
     * @param v
     */

    public void displayVertex( int v )
    {
        System.out.print( vertexList[v].label );
    }

    public int getIndexByLabel( char lable )
    {
        for ( int i = 0; i < vertexList.length; i++ )
        {
            if ( lable == vertexList[i].label )
            {
                return i;
            }
        }

        // 没有这个节点

        return -1;
    }
}

阅读(3737) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~