java編程實(shí)現(xiàn)并查集的路徑壓縮代碼詳解
首先看兩張路徑壓縮的圖片:
并查集(Union-find Sets)是一種非常精巧而實(shí)用的數(shù)據(jù)結(jié)構(gòu),它主要用于處理一些不相交集合的合并問(wèn)題。一些常見(jiàn)的用途有求連通子圖、求最小生成樹(shù)的 Kruskal 算法和求最近公共祖先(Least Common Ancestors, LCA)等。
使用并查集時(shí),首先會(huì)存在一組不相交的動(dòng)態(tài)集合 S={S 1 ,S 2 ,⋯,S k } ,一般都會(huì)使用一個(gè)整數(shù)表示集合中的一個(gè)元素。
每個(gè)集合可能包含一個(gè)或多個(gè)元素,并選出集合中的某個(gè)元素作為代表。每個(gè)集合中具體包含了哪些元素是不關(guān)心的,具體選擇哪個(gè)元素作為代表一般也是不關(guān)心的。我們關(guān)心的是,對(duì)于給定的元素,可以很快的找到這個(gè)元素所在的集合(的代表),以及合并兩個(gè)元素所在的集合,而且這些操作的時(shí)間復(fù)雜度都是常數(shù)級(jí)的。
并查集的基本操作有三個(gè):
makeSet(s):建立一個(gè)新的并查集,其中包含 s 個(gè)單元素集合。
unionSet(x, y):把元素 x 和元素 y 所在的集合合并,要求 x 和 y 所在的集合不相交,如果相交則不合并。
find(x):找到元素 x 所在的集合的代表,該操作也可以用于判斷兩個(gè)元素是否位于同一個(gè)集合,只要將它們各自的代表比較一下就可以了。
package com.dataStructure.union_find;
// 我們的第五版Union-Find
public class UnionFind5 {
  // rank[i]表示以i為根的集合所表示的樹(shù)的層數(shù)
  // 在后續(xù)的代碼中, 我們并不會(huì)維護(hù)rank的語(yǔ)意, 也就是rank的值在路徑壓縮的過(guò)程中, 有可能不在是樹(shù)的層數(shù)值
  // 這也是我們的rank不叫height或者depth的原因, 他只是作為比較的一個(gè)標(biāo)準(zhǔn)
  private int[] rank;
  private int[] parent; // parent[i]表示第i個(gè)元素所指向的父節(jié)點(diǎn)
  private int count;  // 數(shù)據(jù)個(gè)數(shù)
  // 構(gòu)造函數(shù)
  public UnionFind5(int count){
    rank = new int[count];
    parent = new int[count];
    this.count = count;
    // 初始化, 每一個(gè)parent[i]指向自己, 表示每一個(gè)元素自己自成一個(gè)集合
    for( int i = 0 ; i < count ; i ++ ){
      parent[i] = i;
      rank[i] = 1;
    }
  }
  // 查找過(guò)程, 查找元素p所對(duì)應(yīng)的集合編號(hào)
  // O(h)復(fù)雜度, h為樹(shù)的高度
  private int find(int p){
    assert( p >= 0 && p < count );
    // path compression 1
    while( p != parent[p] ){
      parent[p] = parent[parent[p]];
      p = parent[p];
    }
    return p;
    // path compression 2, 遞歸算法
//      if( p != parent[p] )
//        parent[p] = find( parent[p] );
//      return parent[p];
  }
  // 查看元素p和元素q是否所屬一個(gè)集合
  // O(h)復(fù)雜度, h為樹(shù)的高度
  public boolean isConnected( int p , int q ){
    return find(p) == find(q);
  }
  // 合并元素p和元素q所屬的集合
  // O(h)復(fù)雜度, h為樹(shù)的高度
  public void unionElements(int p, int q){
    int pRoot = find(p);
    int qRoot = find(q);
    if( pRoot == qRoot )
      return;
    // 根據(jù)兩個(gè)元素所在樹(shù)的元素個(gè)數(shù)不同判斷合并方向
    // 將元素個(gè)數(shù)少的集合合并到元素個(gè)數(shù)多的集合上
    if( rank[pRoot] < rank[qRoot] ){
      parent[pRoot] = qRoot;
    }
    else if( rank[qRoot] < rank[pRoot]){
      parent[qRoot] = pRoot;
    }
    else{ // rank[pRoot] == rank[qRoot]
      parent[pRoot] = qRoot;
      rank[qRoot] += 1;  // 此時(shí), 我維護(hù)rank的值
    }
  }
}
總結(jié)
以上就是本文關(guān)于java編程實(shí)現(xiàn)并查集的路徑壓縮代碼詳解的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
上一篇:Java編程實(shí)現(xiàn)打印螺旋矩陣實(shí)例代碼
欄 目:Java編程
本文標(biāo)題:java編程實(shí)現(xiàn)并查集的路徑壓縮代碼詳解
本文地址:http://www.jygsgssxh.com/a1/Javabiancheng/8398.html
您可能感興趣的文章
- 01-10Java咖啡館(1)——嘆咖啡
 - 01-10Java Socket編程(三) 服務(wù)器Sockets
 - 01-10Java進(jìn)階:Struts多模塊的技巧
 - 01-10Java Socket編程(一) Socket傳輸模式
 - 01-10Java Socket編程(二) Java面向連接的類
 - 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
 - 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒(méi)有被捕獲的異常
 - 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)器
 - 01-10Java中的浮點(diǎn)數(shù)分析
 - 01-10面向?qū)ο缶幊?Java中的抽象數(shù)據(jù)類型
 


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
 - 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
 - 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
 - 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
 - 5c語(yǔ)言計(jì)算三角形面積代碼
 - 6什么是 WSH(腳本宿主)的詳細(xì)解釋
 - 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
 - 8正則表達(dá)式匹配各種特殊字符
 - 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
 - 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
 
本欄相關(guān)
- 01-10Java咖啡館(1)——嘆咖啡
 - 01-10JVM的垃圾回收機(jī)制詳解和調(diào)優(yōu)
 - 01-10Java Socket編程(三) 服務(wù)器Sockets
 - 01-10Java進(jìn)階:Struts多模塊的技巧
 - 01-10J2SE 1.5版本的新特性一覽
 - 01-10Java Socket編程(一) Socket傳輸模式
 - 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
 - 01-10Java Socket編程(二) Java面向連接的類
 - 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)
 - 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒(méi)有被捕獲的異常
 
隨機(jī)閱讀
- 01-11Mac OSX 打開(kāi)原生自帶讀寫NTFS功能(圖文
 - 04-02jquery與jsp,用jquery
 - 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
 - 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
 - 01-10delphi制作wav文件的方法
 - 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
 - 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
 - 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
 - 01-10C#中split用法實(shí)例總結(jié)
 - 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
 


