java編程實現(xiàn)并查集的路徑壓縮代碼詳解
首先看兩張路徑壓縮的圖片:


并查集(Union-find Sets)是一種非常精巧而實用的數(shù)據(jù)結(jié)構(gòu),它主要用于處理一些不相交集合的合并問題。一些常見的用途有求連通子圖、求最小生成樹的 Kruskal 算法和求最近公共祖先(Least Common Ancestors, LCA)等。
使用并查集時,首先會存在一組不相交的動態(tài)集合 S={S 1 ,S 2 ,⋯,S k } ,一般都會使用一個整數(shù)表示集合中的一個元素。
每個集合可能包含一個或多個元素,并選出集合中的某個元素作為代表。每個集合中具體包含了哪些元素是不關(guān)心的,具體選擇哪個元素作為代表一般也是不關(guān)心的。我們關(guān)心的是,對于給定的元素,可以很快的找到這個元素所在的集合(的代表),以及合并兩個元素所在的集合,而且這些操作的時間復(fù)雜度都是常數(shù)級的。
并查集的基本操作有三個:
makeSet(s):建立一個新的并查集,其中包含 s 個單元素集合。
unionSet(x, y):把元素 x 和元素 y 所在的集合合并,要求 x 和 y 所在的集合不相交,如果相交則不合并。
find(x):找到元素 x 所在的集合的代表,該操作也可以用于判斷兩個元素是否位于同一個集合,只要將它們各自的代表比較一下就可以了。
package com.dataStructure.union_find;
// 我們的第五版Union-Find
public class UnionFind5 {
// rank[i]表示以i為根的集合所表示的樹的層數(shù)
// 在后續(xù)的代碼中, 我們并不會維護rank的語意, 也就是rank的值在路徑壓縮的過程中, 有可能不在是樹的層數(shù)值
// 這也是我們的rank不叫height或者depth的原因, 他只是作為比較的一個標準
private int[] rank;
private int[] parent; // parent[i]表示第i個元素所指向的父節(jié)點
private int count; // 數(shù)據(jù)個數(shù)
// 構(gòu)造函數(shù)
public UnionFind5(int count){
rank = new int[count];
parent = new int[count];
this.count = count;
// 初始化, 每一個parent[i]指向自己, 表示每一個元素自己自成一個集合
for( int i = 0 ; i < count ; i ++ ){
parent[i] = i;
rank[i] = 1;
}
}
// 查找過程, 查找元素p所對應(yīng)的集合編號
// O(h)復(fù)雜度, h為樹的高度
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是否所屬一個集合
// O(h)復(fù)雜度, h為樹的高度
public boolean isConnected( int p , int q ){
return find(p) == find(q);
}
// 合并元素p和元素q所屬的集合
// O(h)復(fù)雜度, h為樹的高度
public void unionElements(int p, int q){
int pRoot = find(p);
int qRoot = find(q);
if( pRoot == qRoot )
return;
// 根據(jù)兩個元素所在樹的元素個數(shù)不同判斷合并方向
// 將元素個數(shù)少的集合合并到元素個數(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; // 此時, 我維護rank的值
}
}
}
總結(jié)
以上就是本文關(guān)于java編程實現(xiàn)并查集的路徑壓縮代碼詳解的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
相關(guān)文章
Java中==與equals()及hashcode()三者之間的關(guān)系詳解
最近也是在讀Hollis的《深入理解Java核心技術(shù)》里面一節(jié)講到了equals()和hashcode()的關(guān)系,對于這個高頻面試點,咱們需要認真理清一下幾者之間的關(guān)系2022-10-10
不喜歡羅里吧嗦,講的很精簡易懂。從基礎(chǔ)開始講,后續(xù)會講到JAVA高級,中間會穿插面試題和項目實戰(zhàn),希望能給大家?guī)韼椭?/div> 2022-03-03
SpringBoot-JPA刪除不成功,只執(zhí)行了查詢語句問題
這篇文章主要介紹了SpringBoot-JPA刪除不成功,只執(zhí)行了查詢語句問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12
SpringBoot之使用Feign實現(xiàn)微服務(wù)間的交互
這篇文章主要介紹了SpringBoot中使用Feign實現(xiàn)微服務(wù)間的交互,對微服務(wù)這方面感興趣的小伙伴可以參考閱讀本文2023-03-03
深入理解SpringBoot?最大連接數(shù)及最大并發(fā)數(shù)
SpringBoot能支持的最大并發(fā)量主要看其對Tomcat的設(shè)置,可以在配置文件中對其進行更改,本文就來介紹一下SpringBoot?最大連接數(shù)及最大并發(fā)數(shù),感興趣的可以了解一下2023-08-08最新評論

