hdoj 3478 Catch(二分图判定+并查集)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3478

思路分析:该问题需要求是否存在某一个时刻,thief可能存在图中没一个点;将该问题转换为图论问题即为判断该图是否为一个连通图且不为二分图;

(1)二分图的性质:对于无向图G=(V, E),如果可以将图中的点划分为两个不相交的点集X与Y = V - X(V为点集),使得图中所有的边邻接的两个点分别存在集合X与集合Y中,则称该图G为二分图;

(2) 二分图判定算法:二分图一种判定方法是给图中的每一个点着黑色或者白色,如果图中每条边邻接的两个点的颜色不同,则该图为二分图,所以非联通的图也可以是二分图,即一个无向图是二分图当且仅当其每个联通分量都是二分图;二分图判定算法使用dfs搜索:选定一个起始结点并着以某一种颜色,从该起始结点出发,与其邻接的点如果已经着色并且颜色与该结点相同,则不为二分图,如果与其邻接的点没有着色,则对该点着以不同的颜色,如此处理直到所有的点被着色成功则该结点所在的联通分量为二分图;需要注意的是图中可能有多个联通分量;

代码如下:

#include <cstdio>
#include <cstring>
#include <vector>
#include <iostream>
using namespace std;

const int MAX_N = 100000 + 100;
vector<int> G[MAX_N];
int fa[MAX_N], color[MAX_N];

void Init()
{
    for (int i = 0; i < MAX_N; ++i)
    {
        fa[i] = i;
        G[i].clear( );
    }
    memset(color, 0, sizeof(color));
}

int Find(int v)
{
    if (fa[v] == v)
        return v;
    else
        return fa[v] = Find(fa[v]);
}

int Union(int a, int b)
{
    int fa_a = Find(a);
    int fa_b = Find(b);

    if (fa_a == fa_b)
        return -1;
    else if (fa_a > fa_b)
        fa[fa_b] = fa_a;
    else
        fa[fa_a] = fa_b;
    return 1;
}

bool BiPartite(int u)
{
    for (int i = 0; i < G[u].size(); ++i)
    {
        int v = G[u][i];
        if (color[u] == color[v])
            return false;
        if (!color[v])
        {
            color[v] = 3 - color[u];
            if (!BiPartite(v))
                return false;
        }
    }
    return true;
}

int main()
{
    int test_case, ver_1, ver_2;
    int ver_num, road_num, start, case_id = 0;

    scanf("%d", &test_case);
    while (test_case--)
    {
        Init();
        scanf("%d %d %d", &ver_num, &road_num, &start);
        for (int i = 0; i < road_num; ++i)
        {
            scanf("%d %d", &ver_1, &ver_2);
            G[ver_1].push_back(ver_2);
            G[ver_2].push_back(ver_1);
            Union(ver_1, ver_2);
        }

        int set_count = 0;
        for (int i = 0; i < ver_num; ++i)
        {
            if (fa[i] == i)
                set_count++;
            if (set_count > 1)
                break;
        }
        printf("Case %d: ", ++case_id);
        color[start] = 1;
        if (!BiPartite(start) && set_count == 1)
            printf("YES\n");
        else
            printf("NO\n");
    }
    return 0;
}
时间: 2024-11-05 17:34:04

hdoj 3478 Catch(二分图判定+并查集)的相关文章

Codeforces 776D.The Door Problem (dfs二分图判定 / 并查集)

题目链接: http://codeforces.com/problemset/problem/776/D 题意: n扇门,m个开关(n,m<=1e5),每个开关控制若干个门,反转开关门状态变化,每个门正好被两个开关控制,问是否有可能把所有门的状态置为1? 思路: from: http://blog.csdn.net/jeremy1149/article/details/56839453 方法一:二分图染色 注意到一个门只被两个开关控制 若门初始为0 则控制它的两个开关状态相反,门初始为1则 要求

poj1703 Find them,Catch them 【并查集】

做过一些的带权并查集,再来做所谓的"种类并查集",发现好像就顿悟了. 种类并查集与带权并查集实质上的差别并不大, 关键的区别就是种类并查集只是带权并查集再弄个%取余操作而已,然后余数就表示他属于哪个种类. 这题只有两个种类,也就是只有0和1两种, 对于两个不同的种类,那么之间的权值是相差1的,所以按照带权并查集的方法做加上1,然后取余2即可. #include<cstdio> const int N = 100005; int n, m, f[N], rank[N]; in

BZOJ 4025 二分图 分治+并查集

题目大意:给定一张n个点的图,有m条边,T个时间段,每条边只存在于(st,ed]这些时间段,求每个时间段内这个图是否是二分图 分治并查集大法好 定义Solve(x,y,E)为当前处理的区间为[x,y],E为所有存在时间为[x,y]的子集的边的集合 那么对于E中的每一条边(u,v),讨论: 若当前边的存在时间为[x,y],则在并查集上判断是否出现奇环 如果出现,[x,y]内的所有时刻都一定不是二分图,输出答案即可 如果不出现,在并查集中连接(u,v) 否则判断存在时间和mid的关系讨论扔进左区间还

poj1703Find them, Catch them(并查集以及路径压缩)

1 /* 2 题目大意:有两个不同的黑帮,开始的时候不清楚每个人是属于哪个的! 3 执行两个操作 4 A a, b回答a, b两个人是否在同一帮派,或者不确定 5 D a, b表示a, b两个人不在同一个帮派 6 7 思路:利用并查集将相同帮派的人合并到一起! 8 a ,b 不在同一个城市,那么 a, 和mark[b]就在同一个城市, 9 b 和 mark[a]就在同一个城市! 10 */ 11 #include<iostream> 12 #include<cstring> 13

HDU-3081-Marriage Match II 二分图匹配+并查集 OR 二分+最大流

二分+最大流: 1 //题目大意:有编号为1~n的女生和1~n的男生配对 2 // 3 //首先输入m组,a,b表示编号为a的女生没有和编号为b的男生吵过架 4 // 5 //然后输入f组,c,d表示编号为c的女生和编号为d的女生是朋友 6 // 7 //进行配对的要求满足其一即可. 8 //1.a女生没有和b男生吵过架 9 //2.a女生的朋友和b男生没有吵过架 10 // 11 //每进行一轮之后重新配对,配过得一对不可再配,问最多能进行几轮. 12 // 13 //题解: 14 //这一道

HDU 3081:Marriage Match II(二分图匹配+并查集)

http://acm.hdu.edu.cn/showproblem.php?pid=3081 题意:有n个男生n个女生,他们只有没有争吵或者女生a与男生A没有争吵,且女生b与女生a是朋友,因此女生b也可以和男生A过家家(具有传递性).给出m个关系,代表女生a和男生b没有争吵过.给出k个关系,代表女生a与女生b是好朋友.每一轮过家家之后,女生只能选择可以选择并且没选过的男生过家家,问游戏能进行几轮. 思路:因为n<=100,因此支持O(n^3)的算法,挺容易想到是一个二分图匹配的.(出现在我的网络

[BZOJ1854][Scoi2010]游戏(二分图匹配/并查集)

题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1854 分析:很裸的一道二分图匹配对吧,但是在hzwer的blog上看见神奇的并查集做法! 其实这题和bzoj1191惊人的相似! 把权值当作点,装备当作边,既然一个装备只能选择一个属性,也就是你要人为给边定向,箭头指到的点就表示你这个装备选的属性. 然后就和bzoj1191一样的了 如果一个集合里的点构成一个树,那么很显然,把那个权值最大的点作为根,除了根节点以外的点都可以选出,如果

POJ 1308 Is It A Tree? &amp;&amp; NYOJ 129 (树的判定+并查集)

[题目链接]click here~~ [题目大意]给定多对节点,判断所有节点能否组成一棵树 [解题思路]并查集的基本操作,定义node,edge,统计node和edge的数目,如果(edge==node-1||node==0)则可以成树 树的判定:n个节点,最多n-1条环,只有一个入度为边,不成0 的点,其他入度不大于1,不过要注意poj数据里如果1 1 0 0也会不符合要求,也就是不能自己指向自己 代码: /* Author:HRW 树的判定: n个节点,最多n-1条边 不成环 只有一个入度为

poj 1703 Find them, Catch them 【并查集拓展】

Find them, Catch them Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 32514   Accepted: 10036 Description The police office in Tadu City decides to say ends to the chaos, as launch actions to root up the TWO gangs in the city, Gang Drago