代碼隨想錄算法訓(xùn)練營(yíng)第55天 | 第十一章 圖論part05:并查集理論基礎(chǔ)恐锦、尋找存在的路徑

第十一章 圖論part05

并查集理論基礎(chǔ)

并查集理論基礎(chǔ)很重要滑燃,明確并查集解決什么問題役听,代碼如何寫,對(duì)后面做并查集類題目很有幫助表窘。
文章講解

并查集的 Java 代碼模板

import java.util.Arrays;

public class UnionFind {
    private int[] parent;  // 存儲(chǔ)每個(gè)節(jié)點(diǎn)的父節(jié)點(diǎn)
    private int n;  // 節(jié)點(diǎn)數(shù)量

    // 構(gòu)造函數(shù)典予,初始化并查集
    public UnionFind(int n) {
        this.n = n;
        parent = new int[n];
        init();
    }

    // 并查集初始化
    private void init() {
        for (int i = 0; i < n; i++) {
            parent[i] = i;
        }
    }

    // 尋找根節(jié)點(diǎn),帶路徑壓縮
    public int find(int u) {
        if (u == parent[u]) {
            return u;
        } else {
            parent[u] = find(parent[u]);
            return parent[u];
        }
    }

    // 判斷兩個(gè)節(jié)點(diǎn)是否在同一個(gè)集合
    public boolean isSame(int u, int v) {
        return find(u) == find(v);
    }

    // 將兩個(gè)節(jié)點(diǎn)加入到同一個(gè)集合
    public void join(int u, int v) {
        int rootU = find(u);
        int rootV = find(v);
        if (rootU != rootV) {
            parent[rootV] = rootU;
        }
    }

    // 測(cè)試用例
    public static void main(String[] args) {
        int n = 1005;  // 根據(jù)具體問題確定節(jié)點(diǎn)數(shù)量
        UnionFind uf = new UnionFind(n);

        // 示例操作
        uf.join(1, 2);
        uf.join(3, 4);
        System.out.println(uf.isSame(1, 2));  // true
        System.out.println(uf.isSame(1, 3));  // false
        uf.join(2, 3);
        System.out.println(uf.isSame(1, 3));  // true
    }
}

代碼解釋:

  1. 初始化 (init):將每個(gè)節(jié)點(diǎn)初始化為自身的父節(jié)點(diǎn)乐严,表示每個(gè)節(jié)點(diǎn)開始時(shí)都是獨(dú)立的集合熙参。
  2. 尋找根節(jié)點(diǎn) (find):使用路徑壓縮技術(shù)來加速后續(xù)查詢。在查找過程中麦备,將訪問過的節(jié)點(diǎn)直接鏈接到根節(jié)點(diǎn)孽椰。
  3. 判斷是否在同一集合 (isSame):通過比較兩個(gè)節(jié)點(diǎn)的根節(jié)點(diǎn)來判斷它們是否在同一個(gè)集合中昭娩。
  4. 將兩個(gè)節(jié)點(diǎn)加入同一集合 (join):將一個(gè)節(jié)點(diǎn)的根節(jié)點(diǎn)鏈接到另一個(gè)節(jié)點(diǎn)的根節(jié)點(diǎn),從而將兩個(gè)集合合并黍匾。

按秩(rank)合并
二刷再看


尋找存在的路徑

并查集裸題栏渺,學(xué)會(huì)理論基礎(chǔ)后,本題直接可以直接刷過
文章講解

import java.util.Scanner;

public class Main {

    static private int[] father; // 按照節(jié)點(diǎn)大小定義數(shù)組大小

    // 并查集初始化
    static public void init(int n) {
        father = new int[n + 1];
        for (int i = 1; i <= n; i++) {
            father[i] = i;
        }
    }

    // 并查集里尋根的過程
    static public int find(int u) {
        if (u == father[u]) {
            return u;
        } else {
            father[u] = find(father[u]); // 路徑壓縮
            return father[u];
        }
    }

    // 判斷 u 和 v 是否找到同一個(gè)根
    static public boolean isSame(int u, int v) {
        u = find(u);
        v = find(v);
        return u == v;
    }

    // 將 v -> u 這條邊加入并查集
    static public void join(int u, int v) {
        u = find(u); // 尋找 u 的根
        v = find(v); // 尋找 v 的根
        if (u == v) return; // 如果發(fā)現(xiàn)根相同锐涯,則說明在一個(gè)集合磕诊,不用兩個(gè)節(jié)點(diǎn)相連直接返回
        father[v] = u;
    }

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        int m = sc.nextInt();
        init(n);// 初始化并查集

        // 處理每一條邊,將其加入并查集
        while (m-- > 0) {
            int s = sc.nextInt();
            int t = sc.nextInt();
            join(s, t);
        }

        // 讀取源節(jié)點(diǎn)和目標(biāo)節(jié)點(diǎn)
        int source = sc.nextInt();
        int destination = sc.nextInt();

        // 判斷源節(jié)點(diǎn)和目標(biāo)節(jié)點(diǎn)是否屬于同一集合
        if (isSame(source, destination)) {
            System.out.println(1);
        } else {
            System.out.println(0);
        }

    }
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末纹腌,一起剝皮案震驚了整個(gè)濱河市霎终,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌升薯,老刑警劉巖莱褒,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異涎劈,居然都是意外死亡广凸,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門蛛枚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來谅海,“玉大人,你說我怎么就攤上這事蹦浦∨び酰” “怎么了?”我有些...
    開封第一講書人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵盲镶,是天一觀的道長(zhǎng)智末。 經(jīng)常有香客問我,道長(zhǎng)徒河,這世上最難降的妖魔是什么系馆? 我笑而不...
    開封第一講書人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮顽照,結(jié)果婚禮上由蘑,老公的妹妹穿的比我還像新娘。我一直安慰自己代兵,他們只是感情好尼酿,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著植影,像睡著了一般裳擎。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上思币,一...
    開封第一講書人閱讀 51,573評(píng)論 1 305
  • 那天鹿响,我揣著相機(jī)與錄音羡微,去河邊找鬼。 笑死惶我,一個(gè)胖子當(dāng)著我的面吹牛妈倔,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播绸贡,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼盯蝴,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了听怕?” 一聲冷哼從身側(cè)響起捧挺,我...
    開封第一講書人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎尿瞭,沒想到半個(gè)月后闽烙,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡筷厘,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年鸣峭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了宏所。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片酥艳。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖爬骤,靈堂內(nèi)的尸體忽然破棺而出充石,到底是詐尸還是另有隱情,我是刑警寧澤霞玄,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布骤铃,位于F島的核電站,受9級(jí)特大地震影響坷剧,放射性物質(zhì)發(fā)生泄漏惰爬。R本人自食惡果不足惜愕提,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一熄攘、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧呐馆,春花似錦狞尔、人聲如沸丛版。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽页畦。三九已至,卻和暖如春研儒,著一層夾襖步出監(jiān)牢的瞬間豫缨,已是汗流浹背独令。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留州胳,地道東北人记焊。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像栓撞,于是被迫代替她去往敵國(guó)和親遍膜。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

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