题目链接

886. 可能的二分法

题意:

给定一组 N 人(编号为 1, 2, ..., N), 我们想把每个人分进任意大小的两组。

每个人都可能不喜欢其他人,那么他们不应该属于同一组。

形式上,如果 dislikes[i] = [a, b],表示不允许将编号为 a 和 b 的人归入同一组。

当可以用这种方法将所有人分进两组时,返回 true;否则返回 false。

思路:

  1. 法一:DFS+染色

    首先初始化所有节点的颜色为0,然后给没有染色的节点染上1,给这个节点的相邻节点染色上-1,以此类推,判断相邻的两个节点颜色相同即可

  2. 法二:种类并查集

    如果a和b不能是相同颜色,那么我们可以合并(a,b+N),(b,a+N)为相同颜色,这样每次判断fa[a]和fa[b]相同即可

class Solution {
public:
//方法二:种类并查集
vector<int> fa;
int findroot(int x){
if(fa[x]==x) return x;
return fa[x]=findroot(fa[x]);
}
void merge(int a,int b){
a=findroot(a);
b=findroot(b);
fa[a]=b;
}
bool possibleBipartition(int N, vector<vector<int>>& dislikes){
fa=vector<int>(N*2+1,0);//开两倍空间(a,b+N),(b,a+N)来存
//初始化fa[]数组
for(int i=1;i<=2*N;i++) fa[i]=i; for(vector<int> tmp:dislikes){
int x=findroot(tmp[0]);
int y=findroot(tmp[1]);
if(x==y) return false;
//合并(a,b+N)
merge(tmp[0],tmp[1]+N);
merge(tmp[1],tmp[0]+N);
}
return true;
} //方法一:
//DFS + 染色
// vector<int> col;//染色
// vector<unordered_set<int>> ed;
// bool dfs(int c,int k){
// col[k]=c;
// for(auto it=ed[k].begin();it!=ed[k].end();it++){
// if(col[*it]==col[k]) return false;
// if(!col[*it] && !dfs(-1*c,*it)) return false;
// }
// return true;
// }
// bool possibleBipartition(int N, vector<vector<int>>& dislikes){
// col=vector<int>(N+1,0);//N个节点的颜色
// ed=vector<unordered_set<int>>(N+1); // for(vector<int> tmp:dislikes){
// ed[tmp[0]].insert(tmp[1]);
// ed[tmp[1]].insert(tmp[0]);//创立邻接链表
// } // for(int i=1;i<=N;i++){
// if(!col[i] && !dfs(1,i)) return false;
// } // return true;
// } // static bool cmp(vector<int> a,vector<int> b){
// if(a[0]==b[0]) return a[1]<b[1];
// return a[0]<b[0];
// }
// bool possibleBipartition(int N, vector<vector<int>>& dislikes) {
// vector<vector<int>> ans;
// for(vector<int> tmp:dislikes){
// if(tmp[0]>tmp[1]) swap(tmp[1],tmp[0]);
// ans.push_back(tmp);
// } // sort(ans.begin(),ans.end());
// set<int> s1,s2;
// for(vector<int> tmp:ans){
// if(s1.count(tmp[1]) || s2.count(tmp[0])) return false;
// s1.insert(tmp[0]);
// s2.insert(tmp[1]);
// }
// return true;
// }
};

leetcode 886. 可能的二分法(DFS,染色,种类并查集)的更多相关文章

  1. NOIP2010关押罪犯[并查集|二分答案+二分图染色 | 种类并查集]

    题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用“怨气值”(一个正整数值)来表示 ...

  2. HDU 5285 wyh2000 and pupil(dfs或种类并查集)

    wyh2000 and pupil Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Other ...

  3. poj 2492 A Bug's Life 二分图染色 || 种类并查集

    题目链接 题意 有一种\(bug\),所有的交往只在异性间发生.现给出所有的交往列表,问是否有可疑的\(bug\)(进行同性交往). 思路 法一:种类并查集 参考:https://www.2cto.c ...

  4. hdu 5313 Bipartite Graph(dfs染色 或者 并查集)

    Problem Description Soda has a bipartite graph with n vertices and m undirected edges. Now he wants ...

  5. A Bug's Life(种类并查集)(也是可以用dfs做)

    http://acm.hdu.edu.cn/showproblem.php?pid=1829   A Bug's Life Time Limit:5000MS     Memory Limit:327 ...

  6. LightOJ 1009 二分图染色+BFS/种类并查集

    题意:有两个阵营的人,他们互相敌对,给出互相敌对的人,问同个阵营的人最多有多少个. 思路:可以使用种类并查集写.也可以使用使用二分图染色的写法,由于给定的点并不是连续的,所以排序离散化一下,再进行BF ...

  7. 洛谷 P1525 关押罪犯 & [NOIP2010提高组](贪心,种类并查集)

    传送门 解题思路 很显然,为了让最大值最小,肯定就是从大到小枚举,让他们分在两个监狱中,第一个不符合的就是答案. 怎样判断是否在一个监狱中呢? 很显然,就是用种类并查集. 种类并查集的讲解——团伙(很 ...

  8. NOI2001|POJ1182食物链[种类并查集 向量]

    食物链 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 65430   Accepted: 19283 Description ...

  9. POJ1703Find them, Catch them[种类并查集]

    Find them, Catch them Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 42416   Accepted: ...

随机推荐

  1. Java静态方法和非静态方法之间的关系

    非静态方法 public class Demo2 {    public static void main(String[] args) {        //实例化这个类 new       //对 ...

  2. 【面试专栏】ArrayList 非线程安全案例并提供三种解决方案

    1. 复现问题 import java.util.ArrayList; import java.util.List; import java.util.UUID; /** * 复现问题 * * @au ...

  3. Difference between @Bean and @Autowired

    Demo01 1 @SpringBootApplication 2 public class Application { 3 4 @Autowired 5 BookingService booking ...

  4. (转) SQL 中的 NULL 你真的懂了吗?【数据库|SQL】

    注:转载自下面链接 https://blog.csdn.net/lnotime/article/details/104847946 SQL 中的 NULL (译自 NULL Values in SQL ...

  5. vue 事件基本用法

    事件基本用法 事件的函数都定义在VUE实例中的methods中,当然也可以直接写在元素内,但是这并不利于后期的维护,需要注意的是:在methods定义的函数内想要引用插值内容,需要使用this,不然就 ...

  6. 在Linux下下载RPM包

    在Linux下下载RPM包,但是不安装 在工作中经常会遇到离线安装RPM包的情况,下面是下载RPM包的方法 # 使用yum下载RPM包 yum -y install --downloadonly &l ...

  7. springboot使用aspectJ

    添加springboot-aop的starter <dependency> <groupId>org.springframework.boot</groupId> ...

  8. 如何优雅的将Object转换成List

    Main主函数中的 Object obj模拟了List对象.后续的代码首先判断obj是否是List类型,然后使用Class.cast做类型转换. 如果你想使用更方便的方法,可以直接调用下面的函数. p ...

  9. java 反射随记

    记录一下有关 Class 对象的相关方法: 1.获取 Class 对象的三个方法: ⑴ 使用 Class.forName("全限定类名") ,参数是该类的全限定类名,可拓展性强: ...

  10. [LeetCode]160. Intersection of Two Linked Lists判断交叉链表的交点

    方法要记住,和判断是不是交叉链表不一样 方法是将两条链表的路径合并,两个指针分别从a和b走不同路线会在交点处相遇 public ListNode getIntersectionNode(ListNod ...