leetcode886. 可能的二分法

题目

给定一组 n 人(编号为 1, 2, …, n), 我们想把每个人分进任意大小的两组。每个人都可能不喜欢其他人,那么他们不应该属于同一组。

给定整数 n 和数组 dislikes ,其中 dislikes[i] = [ai, bi] ,表示不允许将编号为 ai 和 bi的人归入同一组。当可以用这种方法将所有人分进两组时,返回 true;否则返回 false。

示例 1:
输入:n = 4, dislikes = [[1,2],[1,3],[2,4]] 输出:true 解释:group1 [1,4],
group2 [2,3]
示例 2:
输入:n = 3, dislikes = [[1,2],[1,3],[2,3]] 输出:false
示例 3:
输入:n = 5, dislikes = [[1,2],[2,3],[3,4],[4,5],[1,5]] 输出:false

思路

这道题是一道模板题,图着色算法

图着色是指将特定的颜色按照约束条件(相邻任意两个部分的颜色不能相同)在图中分配的问题

我们可以将题目抽象成一个无向图,相邻节点分别用红色、蓝色进行染色,要求相邻节点颜色不相同
在这里插入图片描述
图我们可以使用邻接矩阵表示,即[1,2][1,3][2,4]可以表示为:

在这里插入图片描述
遍历每一行,得到相邻节点;例如想要找到1的相邻节点,遍历第一行,得到2、3;如果将1染成红色的话,需要将2、3染成蓝色,染色过程如下:
在这里插入图片描述

Code

class Solution {
    public boolean possibleBipartition(int n, int[][] dislikes) {
        // 邻接矩阵
        int[][] matrix = new int[n + 1][n + 1];
        for(int[] item : dislikes){
            matrix[item[0]][item[1]] = 1;
            matrix[item[1]][item[0]] = 1;
        }
        // 相邻节点染色
        int[] record = new int[n + 1];
        for(int i = 1; i <= n; i++){
            // 只看未染色的节点
            if(record[i] == 0 && !dfs(matrix, record, i, 1, n)) return false;
        }
        return true;
    }

    private boolean dfs(int[][] matrix, int[] record, int idx, int color, int n){
        record[idx] = color;
        for(int i = 1; i <= n; i++){
            if(i == idx) continue;
            // 相邻节点, 颜色与当前节点相同
            if(matrix[idx][i] == 1 && record[i] == color) return false; 
            // 相邻节点, 未染色, 先尝试染色(dfs)
            if(matrix[idx][i] == 1 && record[i] == 0 && !dfs(matrix, record, i, -1 * color, n)) return false;
        }

        return true;
    }
}

结果

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值