一個無向連通網絡,去掉一個邊集可以使其變成兩個連通分量則這個邊集就是割集;最小割集當然就權和最小的割集。
可以用最小切割最大流定理:
1.min=MAXINT,確定一個源點
2.枚舉匯點
3.計算最大流,并確定當前源匯的最小割集,若比min小更新min
4.轉到2直到枚舉完畢
5.min即為所求輸出min
不難看出復雜度很高:枚舉匯點要O(n),最短增廣路最大流算法求最大流是O((n^2)m)復雜度,在復雜網絡中O(m)=O(n^2),算法總復雜度就是O(n^5);哪怕采用最高標號預進流算法求最大流O((n^2)(m^0.5)),算法總復雜度也要O(n^4)
所以用網絡流算法求解最小割集復雜度不會低于O(n^4)。
---------
prim算法不僅僅可以求最小生成樹,也可以求“最大生成樹”。最小割集Stoer-Wagner算法就是典型的應用實例。
求解最小割集普遍采用Stoer-Wagner算法,不提供此算法證明和代碼,只提供算法思路:
1.min=MAXINT,固定一個頂點P
2.從點P用“類似”prim的s算法擴展出“最大生成樹”,記錄最后擴展的頂點和最后擴展的邊
3.計算最后擴展到的頂點的切割值(即與此頂點相連的所有邊權和),若比min小更新min
4.合并最后擴展的那條邊的兩個端點為一個頂點(當然他們的邊也要合并,這個好理解吧?)
5.轉到2,合并N-1次后結束
6.min即為所求,輸出min
prim本身復雜度是O(n^2),合并n-1次,算法復雜度即為O(n^3)
如果在prim中加堆優化,復雜度會降為O((n^2)logn)
這個Stoer-Wagner算法可以參見這篇paper(http://docs.google.com/fileview?id=0BwxLvD9mcDNtMjk3MWVkMTAtZjMzNi00ZWE3LTkxYjQtYTQwNzcyZTk3Njk2&hl=en), 其核心思想是迭代縮小規模, 算法基于這樣一個事實:
對于圖中任意兩點s和t, 它們要么屬于最小割的兩個不同集中, 要么屬于同一個集.
如果是后者, 那么合并s和t后并不影響最小割. 基于這么個思想, 如果每次能求出圖中某兩點之間的最小割, 然后更新答案后合并它們再繼續求最小割, 就得到最終答案了. 算法步驟如下:
1. 設最小割cut=INF, 任選一個點s到集合A中, 定義W(A, p)為A中的所有點到A外一點p的權總和.
2. 對剛才選定的s, 更新W(A,p)(該值遞增).
3. 選出A外一點p, 且W(A,p)最大的作為新的s, 若A!=G(V), 則繼續2.
4. 把最后進入A的兩點記為s和t, 用W(A,t)更新cut.
5. 新建頂點u, 邊權w(u, v)=w(s, v)+w(t, v), 刪除頂點s和t, 以及與它們相連的邊.
6. 若|V|!=1則繼續1.
看起來很簡單, 每次像做最大生成樹一樣選最大"邊"(注意, 這里其實不是邊, 而是已經累計的權值之和, 就當是加權的度好了), 然后把最后進入的兩個點縮到一塊就可以了. 合并點最多有n-1次, 而不加堆優化的prim是O(n^2)的, 所以最終復雜度O(n^3), 要是你有心情敲一大坨代碼, 還可以在稀疏圖上用Fibonacci Heap優化一下, 不過網上轉了一圈, 大多都是說能用Fibonacci Heap優化到怎樣怎樣的復雜度, 真正能自己寫出來的恐怕也沒幾個, 看看uoregon(俄勒岡大學)的一大坨代碼就有點寒. (http://resnet.uoregon.edu/~gurney_j/jmpc/fib.html)
特別注意幾個地方, 網上的好幾個Stoer-Wagner版本都存在一些小錯誤:
1. 算法在做"最大生成樹"時更新的不是普通意義上的最大邊, 而是與之相連的邊的權值和, 當所有邊都是單位權值時就是累計度.
2. "最后進入A的兩點記為s和t", 網上對s有兩種解釋, 一是在t之前一個加進去的點, 二是t的前趨節點, 也就是最后選擇的那條邊的另一端. 正解是第一種!
3. 對于稠密圖, 比如這題, 我用堆, 映射二分堆, 或者STL的優先隊列都會TLE, 還不如老老實實O(n^3).