欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java C++題解leetcode886可能的二分法并查集染色法

 更新時間:2022年10月17日 10:23:37   作者:AnjaVon  
這篇文章主要為大家介紹了Java C++題解leetcode886可能的二分法并查集染色法實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

題目要求

思路一:反向點+并查集

  • 根據(jù)題意不喜歡就不在一個組可以想到使用并查集,本題是兩個集合所以對每一個節(jié)點引入一個反向點,使兩者分屬于不同集合,借此記錄前續(xù)節(jié)點維持的不喜歡關(guān)系;
  • 在將每個節(jié)點xxx放入組合時,同時將其反向節(jié)點x+nx+nx+n放入另一組合,然后向后遍歷依次處理每個節(jié)點,同時判斷相互不喜歡的兩個點當前是否會被迫放入一個集合(連通),若是則無法滿足題意。

下面淺學一些并查集的基本概念,然后再去實現(xiàn)思路——

淺學并查集(Union Find)

學習參考鏈接

  • 從介紹到不斷優(yōu)化的整個構(gòu)造推導過程,圖片示例與解釋很清晰。

簡介:

一種樹型的數(shù)據(jù)結(jié)構(gòu),用于處理一些不相交集合的合并查詢問題;

  • 核心思想:
    • 用一個數(shù)組表示整片森林,樹的根節(jié)點唯一標識了一個集合,只要找到了某個元素的樹根,就能確定它在哪個集合里;
  • 適用場景:
    • 用于需要反復查找某一元素屬于哪個集合以進行集合合并的場景,用其他數(shù)據(jù)結(jié)構(gòu)解決該類問題將造成巨大的時空開銷。

基礎(chǔ)操作:

通常包括三個函數(shù)

函數(shù)功能
find(x)查找元素xxx屬于哪個集合,也就是找當前元素所在樹的根節(jié)點,查找的同時進行路徑壓縮
union(a, b)合并元素aaa和元素bbb所屬集合,根據(jù)樹高合并兩棵樹
isConnected(a, b)判斷aaa和元素bbb是否處于同一集合中,也就是判斷二者根是否相同

Java

class Solution {
    int[] p = new int[4010]; // 并查集數(shù)組,存父級節(jié)點
    // 找當前節(jié)點的根
    int find(int x) {
        if(p[x] != x) // 非根節(jié)點
            p[x] = find(p[x]); // 繼續(xù)向下找根并進行路徑壓縮
        return p[x];
    }
    // 連接兩節(jié)點的根
    void union(int a, int b) {
        p[find(a)] = p[find(b)];
    }
    // 兩節(jié)點是否連通
    boolean isConnected(int a, int b) {
        return find(a) == find(b);
    }
    public boolean possibleBipartition(int n, int[][] dislikes) {
        for (int i = 1; i <= 2 * n; i++) // 節(jié)點+反向節(jié)點
            p[i] = i; // 初始化并查集,指向自己
        for (int[] cur : dislikes) {
            int a = cur[0], b = cur[1];
            if (isConnected(a, b)) // 連通,被迫在一組
                return false;
            // 利用反向節(jié)點維護連通關(guān)系
            union(a, b + n);
            union(b, a + n);
        }
        return true;
    }
}
  • 時間復雜度:O(n+m),其中m為dislikes的長度
  • 空間復雜度:O(n)

C++

  • 注意union會和C++中的預(yù)定義函數(shù)重名
class Solution {
public:
    int p[4010]; // 并查集數(shù)組,存父級節(jié)點
    // 找當前節(jié)點的根
    int find(int x) {
        if(p[x] != x) // 非根節(jié)點
            p[x] = find(p[x]); // 繼續(xù)向下找根并進行路徑壓縮
        return p[x];
    }
    // 連接兩節(jié)點的根
    void unionn(int a, int b) {
        p[find(a)] = p[find(b)];
    }
    // 兩節(jié)點是否連通
    bool isConnected(int a, int b) {
        return find(a) == find(b);
    }
    bool possibleBipartition(int n, vector<vector<int>>& dislikes) {
        for (int i = 1; i <= 2 * n; i++) // 節(jié)點+反向節(jié)點
            p[i] = i; // 初始化并查集,指向自己
        for (auto cur : dislikes) {
            int a = cur[0], b = cur[1];
            if (isConnected(a, b)) // 連通,被迫在一組
                return false;
            // 利用反向節(jié)點維護連通關(guān)系
            unionn(a, b + n);
            unionn(b, a + n);
        }
        return true;
    }
};
  • 時間復雜度:O(n+m)
  • 空間復雜度:O(n)

思路二:染色法

  • 將不喜歡數(shù)組存成一個無向圖,給分屬兩個不同集合的點染上不同的顏色,不斷更新染色并判斷不喜歡關(guān)系是否能夠成立;
  • 采用鏈式前向星存儲構(gòu)建無向圖,有邊的兩者不能是同一個顏色,用1和2表示兩種不同的顏色,用000表示未染色;
  • 定義一個DFS(node, clr)函數(shù)表示將節(jié)點node染成clr色

Java

class Solution {
    int N = 2010, M = 2 * 10010;
    int[] head = new int[N], edge = new int[M], next = new int[M];
    int[] color = new int[N];
    int idx = 0;;
    void add(int a, int b) {
        edge[idx] = b;
        next[idx] = head[a];
        head[a] = idx++;
    }
    boolean DFS(int node, int clr) {
        color[node] = clr;
        for (int i = head[node]; i != -1; i = next[i]) {
            int j = edge[i];
             // 不喜歡雙方同色
            if (color[j] == clr)
                return false;
            if (color[j] == 0 && !DFS(j, 3 - clr))
                return false;
        }
        return true;
    }
    public boolean possibleBipartition(int n, int[][] dislikes) {
        Arrays.fill(head, -1);
        for (int[] cur : dislikes) { // 構(gòu)建無向圖
            int a = cur[0], b = cur[1];
            add(a, b);
            add(b, a);
        }
        for (int i = 1; i <= n; i++) {
            if (color[i] != 0) // 已經(jīng)染過
                continue;
            if (!DFS(i, 1)) // 無法染色成功
                return false;
        }
        return true;
    }
}
  • 時間復雜度:O(n+m)
  • 空間復雜度:O(n+m)

C++

class Solution {
public:
    static const int N = 2010, M = 2 * 10010;
    int head[N], edge[M], next[M];
    int color[N];
    int idx = 0;;
    void add(int a, int b) {
        edge[idx] = b;
        next[idx] = head[a];
        head[a] = idx++;
    }
    bool DFS(int node, int clr) {
        color[node] = clr;
        for (int i = head[node]; i != -1; i = next[i]) {
            int j = edge[i];
             // 不喜歡雙方同色
            if (color[j] == clr)
                return false;
            if (color[j] == 0 && !DFS(j, 3 - clr))
                return false;
        }
        return true;
    }
    bool possibleBipartition(int n, vector<vector<int>>& dislikes) {
        memset(head, -1, sizeof(head));
        for (auto cur : dislikes) { // 構(gòu)建無向圖
            int a = cur[0], b = cur[1];
            add(a, b);
            add(b, a);
        }
        for (int i = 1; i <= n; i++) {
            if (color[i] != 0) // 已經(jīng)染過
                continue;
            if (!DFS(i, 1)) // 無法染色成功
                return false;
        }
        return true;
    }
};
  • 時間復雜度:O(n+m)
  • 空間復雜度:O(n+m)

總結(jié)

算法題就回避一下Rust……待我學成歸來……

填了拖了好久的并查集的坑還捎帶復習了一波鏈式前向星存圖;

感覺鏈式前向星忘得差不多了……

以上就是Java C++題解leetcode886可能的二分法并查集染色法的詳細內(nèi)容,更多關(guān)于Java C++ 可能的二分法的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • MybatisPlus調(diào)用原生SQL的實現(xiàn)方法

    MybatisPlus調(diào)用原生SQL的實現(xiàn)方法

    本文主要介紹了MybatisPlus調(diào)用原生SQL的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-02-02
  • JavaScript的基本類型值-String類型

    JavaScript的基本類型值-String類型

    String類型用于表示由零或多個16位Unicode字符組成的字符序列,即字符串。在JavaScript中沒有單個的字符型,都是字符串。這篇文章主要介紹了JavaScript的基本類型值String類型,需要的朋友可以參考下
    2017-02-02
  • 如何通過XML方式配置并實現(xiàn)Mybatis

    如何通過XML方式配置并實現(xiàn)Mybatis

    這篇文章主要介紹了如何通過XML方式配置并實現(xiàn)Mybatis,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-11-11
  • 一文秒懂logstash收集springboot日志的方法

    一文秒懂logstash收集springboot日志的方法

    通過這篇文章帶你了解logstash收集springboot日志的方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04
  • 在Java中實現(xiàn)線程安全的單例模式的常見方式

    在Java中實現(xiàn)線程安全的單例模式的常見方式

    單例模式是一種常用的軟件設(shè)計模式,它確保一個類只有一個實例,并提供一個全局訪問點,在多線程環(huán)境下,確保單例模式的線程安全性是非常重要的,因為多個線程可能會同時嘗試創(chuàng)建實例,導致實例不唯一的問題,本文介紹了在Java中實現(xiàn)線程安全的單例模式有幾種常見的方式
    2024-09-09
  • @RereshScope刷新的原理詳解

    @RereshScope刷新的原理詳解

    在配合配置中心修改配置讓應(yīng)用自動刷新配置時,我們要在需要感知配置變化的bean上面加上@RereshScope。如果我們不加上這注解,那么有可能無法完成配置自動刷新。本文就來和大家講講@RereshScope刷新的原理,需要的可以參考一下
    2022-12-12
  • 如何發(fā)布jar包到maven中央倉庫

    如何發(fā)布jar包到maven中央倉庫

    這篇文章主要介紹了發(fā)布jar包到maven中央倉庫的相關(guān)知識,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2023-12-12
  • springboot如何獲取請求者的ip地址

    springboot如何獲取請求者的ip地址

    在Spring框架中,可以使用攔截器(Interceptor)來監(jiān)聽每個控制器(Controller)的請求,并記錄請求者的IP地址,這篇文章主要介紹了springboot如何獲取請求者的ip地址,需要的朋友可以參考下
    2024-07-07
  • 教你輕松制作java視頻播放器

    教你輕松制作java視頻播放器

    這篇文章主要為大家詳細介紹了如何編寫屬于自己的java視頻播放器,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Java自定義比較器實現(xiàn)中文排序

    Java自定義比較器實現(xiàn)中文排序

    這篇文章主要介紹了Java自定義比較器實現(xiàn)中文排序,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08

最新評論