• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            O(1) 的小樂

            Job Hunting

            公告

            記錄我的生活和工作。。。
            <2012年11月>
            28293031123
            45678910
            11121314151617
            18192021222324
            2526272829301
            2345678

            統(tǒng)計

            • 隨筆 - 182
            • 文章 - 1
            • 評論 - 41
            • 引用 - 0

            留言簿(10)

            隨筆分類(70)

            隨筆檔案(182)

            文章檔案(1)

            如影隨形

            搜索

            •  

            最新隨筆

            最新評論

            閱讀排行榜

            評論排行榜

            有向圖強連通分量 Gabow 算法

             

            When the depth-first search reaches a vertex v, the algorithm performs the following steps:

            1. Set the preorder number of v to C, and increment C.
            2. Push v onto S and also onto P.
            3. For each edge from v to a neighboring vertex w:
              • If the preorder number of w has not yet been assigned, recursively search w;
              • Otherwise, if w has not yet been assigned to a strongly connected component:
                • Repeatedly pop vertices from P until the top element of P has a preorder number less than or equal to the preorder number of w.
            4. If v is the top element of P:
              • Pop vertices from S until v has been popped, and assign the popped vertices to a new component.
              • Pop v from P.

             

            Gabow_Algorithm偽代碼:

            step1:

            找一個沒有被訪問過的節(jié)點v,goto step2(v)。否則,算法結(jié)束。

            step2(v):

            將v壓入堆棧stk1[]和stk2[]

            對于v所有的鄰接頂點u:

            1) 如果沒有訪問過,則step2(u)

            2) 如果訪問過,但沒有刪除,維護stk2[](處理環(huán)的過程,在stk2 中刪除構(gòu)成環(huán)的節(jié)點)

            如果stk2[]的頂元素==v,那么輸出相應(yīng)的強連通分量

             

            這個算法其實就是Tarjan算法的變異體,我們觀察一下,只是它用第二個堆棧來輔助求出強連通分量的根,而不是Tarjan算法里面的DFN[]和Low[]數(shù)組。那么,我們說一下如何使用第二個堆棧來輔助求出強連通分量的根。

            我們使用類比方法,在Tarjan算法中,每次Low[i]的修改都是由于環(huán)的出現(xiàn)(不然,Low[i]的值不可能變小),每次出現(xiàn)環(huán),在這個環(huán)里面只剩下一個Low[i]沒有被改變(深度最低的那個),或者全部被改變,因為那個深度最低的節(jié)點在另一個環(huán)內(nèi)。那么Gabow算 法中的第二堆棧變化就是刪除構(gòu)成環(huán)的節(jié)點,只剩深度最低的節(jié)點,或者全部刪除,這個過程是通過出棧來實現(xiàn),因為深度最低的那個頂點一定比前面的先訪問,那 么只要出棧一直到棧頂那個頂點的訪問時間不大于深度最低的那個頂點。其中每個被彈出的節(jié)點屬于同一個強連通分量。那有人會問:為什么彈出的都是同一個強連 通分量?因為在這個節(jié)點訪問之前,能夠構(gòu)成強連通分量的那些節(jié)點已經(jīng)被彈出了,這個對Tarjan算法有了解的都應(yīng)該清楚,那么Tarjan算法中的判斷根我們用什么來代替呢?想想,其實就是看看第二個堆棧的頂元素是不是當前頂點就可以了。

            現(xiàn)在,你應(yīng)該明白其實Tarjan算法和Gabow算法其實是同一個思想的不同實現(xiàn),但是,Gabow算法更精妙,時間更少(不用頻繁更新Low[])。

             

            QQ截圖未命名

             

            代碼:

            #include "cstdlib"
            #include "cctype"
            #include "cstring"
            #include "cstdio"
            #include "cmath"
            #include "algorithm"
            #include "vector"
            #include "string"
            #include "iostream"
            #include "sstream"
            #include "set"
            #include "queue"
            #include "stack"
            #include "fstream"
            #include "strstream"
            using namespace std;

            #define  M 2000              //題目中可能的最大點數(shù)      
            int STACK[M],top=0;          //Gabow 算法中的輔助棧
            int STACK2[M],top2=0;        //
            int DFN[M];                  //深度優(yōu)先搜索訪問次序
            int ComponetNumber=0;        //有向圖強連通分量個數(shù)
            int Index=0;                 //索引號
            int Belong[M];               //某個點屬于哪個連通分支
            vector <int> Edge[M];        //鄰接表表示
            vector <int> Component[M];   //獲得強連通分量結(jié)果

            void Gabow(int i)
            {
                int j;
                DFN[i]=Index++;
                STACK[++top]=i;
                STACK2[++top2]=i;
                for (int e=0;e<Edge[i].size();e++)
                {
                    j=Edge[i][e];
                    if (DFN[j]==-1)  Gabow(j);
                    else if (Belong[j]==-1)       //如果訪問過,但沒有刪除,維護STACK2
                    {
                        while(DFN[STACK2[top2]]>DFN[j])        //刪除構(gòu)成環(huán)的頂點
                            top2--;
                    }
                }
                if(i==STACK2[top2])               //如果Stack2 的頂元素等于i,輸出相應(yīng)的強連通分量
                {
                    --top2; ++ComponetNumber;
                    do
                    {
                        Belong[STACK[top]]=ComponetNumber;
                        Component[ComponetNumber].push_back(STACK[top]);
                    }while ( STACK[top--]!=i);
                }
            }

            void solve(int N)     //此圖中點的個數(shù),注意是0-indexed!
            {
                memset(STACK,-1,sizeof(STACK));
                memset(STACK2,-1,sizeof(STACK2));
                memset(Belong,-1,sizeof(Belong));
                memset(DFN,-1,sizeof(DFN));

                for(int i=0;i<N;i++)
                    if(DFN[i]==-1)
                        Gabow(i);   
            }
            /*
            此算法正常工作的基礎(chǔ)是圖是0-indexed的。但是獲得的結(jié)果Component數(shù)組和Belong數(shù)組是1-indexed
            */
            int main()
            {
                Edge[0].push_back(1);Edge[0].push_back(2);
                Edge[1].push_back(3);
                Edge[2].push_back(3);Edge[2].push_back(4);
                Edge[3].push_back(0);Edge[3].push_back(5);
                Edge[4].push_back(5);
                int  N=6;
                solve(N);
                cout<<"ComponetNumber is "<<ComponetNumber<<endl;
                for(int i=0;i<N;i++)
                        cout<<Belong[i]<<" ";
                cout<<endl;
                for(int i=0;i<N;i++)
                {
                    for(int j=0;j<Component[i].size();j++)
                        cout<<Component[i][j];
                    cout<<endl;
                }
                return 0;
            }

             

            Reference:

            http://rchardx.is-programmer.com/posts/14898.html

            wiki

             

            終于搞完了SCC問題的3大算法:

            小總結(jié)一下:

             

            三種算法的時間復(fù)雜度都是O(M+N) (N為圖的點數(shù),M為邊數(shù))

             

            Tarjan 算法和Gabow算法思想類似,在Tarjan算法中用Low[]數(shù)組來記錄所能達到的最小時間戳,而Gabow算法則是用Stack2[]輔助獲得了強連通分量的根!

            二者實質(zhì)是一樣的!

             

            Kosaraju 算法則是兩次DFS,所以在時間復(fù)雜度常數(shù)因子的比拼上,肯定拼不過 Tarjan Gabow,但是額外的常數(shù)帶來了良好的拓撲性質(zhì)!它得到的節(jié)點是按照拓撲序組織好的,在求解2-SAT的過程中十分方便。

             

            下面是題目來總結(jié)或者來一個求無向圖雙連通分量Tarjan算法 和求最近公共祖先的離線Tarjan算法!

             

            再引用一次:

            Tarjan算法與求無向圖的雙連通分量(割點、橋)的Tarjan算法也有著很深的聯(lián)系。學(xué)習(xí)該Tarjan算法,也有助于深入理解求雙連通分量的Tarjan算法,兩者可以類比、組合理解。

            求有向圖的強連通分量的Tarjan算法是以其發(fā)明者Robert Tarjan命名的。Robert Tarjan還發(fā)明了求雙連通分量的Tarjan算法,以及求最近公共祖先的離線Tarjan算法,在此對Tarjan表示崇高的敬意。

            posted on 2010-09-27 14:07 Sosi 閱讀(1394) 評論(2)  編輯 收藏 引用

            評論

            # re: 有向圖強連通分量 Gabow 算法 2012-04-18 23:06 劉濤

            您好,我用您的程序求有向圖的強連通分量。但是當點數(shù)很多的時候,總是顯示stack overflow,我的編譯器是vs2008.請問怎么解決這個問題,謝謝。
            您寫的程序真心不錯,謝謝您的饋贈。
              回復(fù)  更多評論    

            # re: 有向圖強連通分量 Gabow 算法 2012-11-07 14:35 ynkdyx@sohu.com

            話說這不就是棧溢出了么,寫的很明白了。如果怕棧溢出就把內(nèi)存開大好了,再用上STL里的vector即可。@劉濤
              回復(fù)  更多評論    

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            統(tǒng)計系統(tǒng)
            99久久综合狠狠综合久久止| 国产精品中文久久久久久久| 99久久99久久精品免费看蜜桃| 久久精品水蜜桃av综合天堂| 狠狠色丁香婷综合久久| 94久久国产乱子伦精品免费| 人妻无码精品久久亚瑟影视| 中文字幕日本人妻久久久免费| 精品久久久久久亚洲| 久久99国产精品久久99小说 | 久久久久成人精品无码| 青青草原综合久久大伊人导航| 性做久久久久久久| 精品久久久久久无码免费| 青青草原精品99久久精品66| 久久噜噜久久久精品66| 久久久久成人精品无码中文字幕| 久久精品国产亚洲Aⅴ香蕉| 久久国产亚洲高清观看| 四虎影视久久久免费观看| 91精品国产91久久| 久久棈精品久久久久久噜噜| 久久一区二区三区99| 岛国搬运www久久| 久久精品国产亚洲综合色| 无码人妻久久一区二区三区| 亚洲精品高清一二区久久| 91久久福利国产成人精品| 97久久国产亚洲精品超碰热 | 久久99精品国产自在现线小黄鸭| 热RE99久久精品国产66热| 国产成人久久精品麻豆一区| 久久精品国产亚洲网站| 国产精品久久国产精麻豆99网站| 久久精品国产久精国产果冻传媒| 国产精品成人久久久| 亚州日韩精品专区久久久| 少妇久久久久久被弄到高潮| 久久久久99精品成人片三人毛片 | 久久亚洲中文字幕精品一区| 日韩美女18网站久久精品|