圖的拓?fù)渑判?/h1>

相關(guān)概念

AOV網(wǎng):在一個(gè)表示工程的有向圖中,用頂點(diǎn)表示活動(dòng),用弧表示活動(dòng)之間的優(yōu)先關(guān)系,這樣的有向圖為頂點(diǎn)表示活動(dòng)的網(wǎng),稱為AOV網(wǎng)(Activity On Vertex)。
AVO網(wǎng)不存在環(huán)路

拓?fù)湫蛄校涸O(shè)G=(V,E)是一個(gè)具有n個(gè)頂點(diǎn)的有向圖,V中頂點(diǎn)序列V1,V2,......,Vn,滿足若從頂點(diǎn)Vi到Vj有一條路徑,則在頂點(diǎn)序列中頂點(diǎn)Vi必在頂點(diǎn)Vj之前,則這樣的頂點(diǎn)序列稱為一個(gè)拓?fù)湫蛄小?br> 拓?fù)湫蛄胁⒉晃ㄒ?/strong>

拓?fù)渑判蚓褪菢?gòu)造拓?fù)湫蛄械倪^程,當(dāng)AOV網(wǎng)中不存在環(huán)路時(shí),全部頂點(diǎn)都會(huì)被輸出。

拓?fù)渑判蛩惴?/h1>

思想:從AOV網(wǎng)中選擇一個(gè)入度為0的頂點(diǎn)輸出,然后刪除此頂點(diǎn),并刪除一次頂點(diǎn)為尾的弧,繼續(xù)重復(fù)該步驟,直至輸出全部頂點(diǎn)或者AOV網(wǎng)中不存在入度為0的頂點(diǎn)為止。

由于拓?fù)渑判蛐枰獎(jiǎng)h除頂點(diǎn),所以使用鄰接表的方式存儲(chǔ)圖會(huì)較為方便

鄰接表結(jié)構(gòu)

鄰接表結(jié)構(gòu)圖.png

鄰接表的結(jié)構(gòu)不局限于此,可以根據(jù)實(shí)際情況添加字段,如在拓?fù)渑判蛑锌梢栽陧旤c(diǎn)表中增加入度字段,用于統(tǒng)計(jì)每個(gè)頂點(diǎn)的入度情況。在帶權(quán)圖中可以在邊表中添加weight字段,用于表示每條邊的權(quán)值。

測試圖:


測試圖.png

對應(yīng)的鄰接表結(jié)構(gòu):


鄰接表.png

源代碼

頂點(diǎn)表類

/**
 * 頂點(diǎn)表結(jié)點(diǎn)類
 * 
 * @author Shaw
 *
 */
class VertexNode {
    // 頂點(diǎn)入度
    private int in;
    // 頂點(diǎn)域
    private String data;
    // 指向邊表
    private EdgeNode firstEdge;

    public VertexNode(int in, String data, EdgeNode firstEdge) {
        super();
        this.in = in;
        this.data = data;
        this.firstEdge = firstEdge;
    }

    public int getIn() {
        return in;
    }

    public void setIn(int in) {
        this.in = in;
    }

    public String getData() {
        return data;
    }

    public void setData(String data) {
        this.data = data;
    }

    public EdgeNode getFirstEdge() {
        return firstEdge;
    }

    public void setFirstEdge(EdgeNode firstEdge) {
        this.firstEdge = firstEdge;
    }
}

邊集表類:

/**
 * 拓?fù)渑判? * 
 * @author Shaw
 *
 */
public class Topological {
    private List<VertexNode> verList;

    //建圖
    public void buildAovGraph() {
        VertexNode v0 = new VertexNode(0, "v0", null);

        EdgeNode v0e1 = new EdgeNode(11, 0, null);
        EdgeNode v0e2 = new EdgeNode(5, 0, null);
        EdgeNode v0e3 = new EdgeNode(4, 0, null);

        v0.setFirstEdge(v0e1);
        v0e1.setNext(v0e2);
        v0e2.setNext(v0e3);

        VertexNode v1 = new VertexNode(0, "v1", null);

        EdgeNode v1e1 = new EdgeNode(8, 0, null);
        EdgeNode v1e2 = new EdgeNode(4, 0, null);
        EdgeNode v1e3 = new EdgeNode(2, 0, null);

        v1.setFirstEdge(v1e1);
        v1e1.setNext(v1e2);
        v1e2.setNext(v1e3);

        VertexNode v2 = new VertexNode(0, "v2", null);

        EdgeNode v2e1 = new EdgeNode(9, 0, null);
        EdgeNode v2e2 = new EdgeNode(6, 0, null);
        EdgeNode v2e3 = new EdgeNode(5, 0, null);

        v2.setFirstEdge(v2e1);
        v2e1.setNext(v2e2);
        v2e2.setNext(v2e3);

        VertexNode v3 = new VertexNode(0, "v3", null);

        EdgeNode v3e1 = new EdgeNode(13, 0, null);
        EdgeNode v3e2 = new EdgeNode(2, 0, null);

        v3.setFirstEdge(v3e1);
        v3e1.setNext(v3e2);

        VertexNode v4 = new VertexNode(0, "v4", null);

        EdgeNode v4e1 = new EdgeNode(7, 0, null);

        v4.setFirstEdge(v4e1);

        VertexNode v5 = new VertexNode(0, "v5", null);

        EdgeNode v5e1 = new EdgeNode(12, 0, null);
        EdgeNode v5e2 = new EdgeNode(8, 0, null);

        v5.setFirstEdge(v5e1);
        v5e1.setNext(v5e2);

        VertexNode v6 = new VertexNode(0, "v6", null);

        EdgeNode v6e1 = new EdgeNode(5, 0, null);

        v6.setFirstEdge(v6e1);

        VertexNode v7 = new VertexNode(0, "v7", null);

        VertexNode v8 = new VertexNode(0, "v8", null);

        EdgeNode v8e1 = new EdgeNode(7, 0, null);

        v8.setFirstEdge(v8e1);

        VertexNode v9 = new VertexNode(0, "v9", null);

        EdgeNode v9e1 = new EdgeNode(11, 0, null);
        EdgeNode v9e2 = new EdgeNode(10, 0, null);

        v9.setFirstEdge(v9e1);
        v9e1.setNext(v9e2);

        VertexNode v10 = new VertexNode(0, "v10", null);

        EdgeNode v10e1 = new EdgeNode(13, 0, null);

        v10.setFirstEdge(v10e1);

        VertexNode v11 = new VertexNode(0, "v11", null);

        VertexNode v12 = new VertexNode(0, "v12", null);

        EdgeNode v12e1 = new EdgeNode(9, 0, null);

        v12.setFirstEdge(v12e1);

        VertexNode v13 = new VertexNode(0, "v13", null);

        verList = new ArrayList<>();
        verList.add(v0);
        verList.add(v1);
        verList.add(v2);
        verList.add(v3);
        verList.add(v4);
        verList.add(v5);
        verList.add(v6);
        verList.add(v7);
        verList.add(v8);
        verList.add(v9);
        verList.add(v10);
        verList.add(v11);
        verList.add(v12);
        verList.add(v13);

    }

    public void topological_sort() {
        Stack<Integer> stack = new Stack<>();
        int count = 0;
        // 初始化所有頂點(diǎn)表結(jié)點(diǎn)的入度值
        for (int i = 0; i < verList.size(); i++) {
            EdgeNode edgeNode = verList.get(i).getFirstEdge();
            while (edgeNode != null) {
                //邊集表中出現(xiàn)的下標(biāo)(adjvex)所對應(yīng)的頂點(diǎn)入度加1
                VertexNode vertexNode = verList.get(edgeNode.getAdjvex());
                vertexNode.setIn(vertexNode.getIn() + 1);
                edgeNode = edgeNode.getNext();
            }
        }
        // 將所有入度為0的頂點(diǎn)入棧
        for (int i = 0; i < verList.size(); i++) {
            if (verList.get(i).getIn() == 0) {
                stack.push(i);
            }
        }

        while (!stack.isEmpty()) {
            // 從棧中彈出一個(gè)入度為0的頂點(diǎn)
            int vertex = stack.pop();
            System.out.print(verList.get(vertex).getData() + " -> ");
            count++;
            // 獲取彈出的頂點(diǎn)指向的第一個(gè)邊結(jié)點(diǎn)
            EdgeNode edgeNode = verList.get(vertex).getFirstEdge();
            while (edgeNode != null) {
                // 獲取邊表結(jié)點(diǎn)的Adjvex值,該值存儲(chǔ)對應(yīng)頂點(diǎn)表結(jié)點(diǎn)的下標(biāo)值
                int adjvex = edgeNode.getAdjvex();
                // 獲取邊表結(jié)點(diǎn)指向的頂點(diǎn)表結(jié)點(diǎn)
                VertexNode vertexNode = verList.get(adjvex);
                // 刪除邊結(jié)點(diǎn),也就是將對應(yīng)的頂點(diǎn)表結(jié)點(diǎn)的入度減1
                vertexNode.setIn(vertexNode.getIn() - 1);
                // 如果該頂點(diǎn)表結(jié)點(diǎn)入度為0時(shí)壓棧
                if (vertexNode.getIn() == 0) {
                    stack.push(adjvex);
                }
                // 獲取下一個(gè)邊表結(jié)點(diǎn)的值
                edgeNode = edgeNode.getNext();
            }
        }
    }
}

測試程序:

public class TopologicalMain {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Topological topological = new Topological();
        topological.buildAovGraph();
        topological.topological_sort();
    }

}

測試結(jié)果:


拓?fù)渑判蚪Y(jié)果圖.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,002評論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,400評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,136評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,714評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,452評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,818評論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,812評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,997評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,552評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,292評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,510評論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,035評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,721評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,121評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,429評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,235評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,480評論 2 379

推薦閱讀更多精彩內(nèi)容