POJ-1182-种类并查集

第一次写合并函数和判断假话的条件不熟练。模板题。

------------------------------------------------------------------------------------------------更新

上面这句话是大一写的。。。确实是模版题,但是遇到种类并查集后并不是很理解了,看了一通回来写写。

首先并查集大致分为,带权并查集、种类并查集、可逆并查集和可持续化并查集。

这个题是最基础的种类并查集,一般的并查集就是区分一下几个东西是不是一个类,而种类并查集是维护一个特定的各个种类之间的关系,不但需要判断是不是同类还能判断几个种类之间的关系。

以这个题来说,我们把数组开3*n的大小,其中1-n表示a种类,n+1-n*2表示b种类,2*n+1-n*3表示c种类。

然后根据输入的关系来维护。

比如输入一句真话x,y同类,那么就合并x,y和x+n,y+n和x+2*n,y+2*n,表示不管x,y是哪类,他俩必定同类。

再比如输入一句真话x,y是吃与被吃的关系,那么就合并x,y+n和x+n,y+2*n和x+2*n,y,表示如果x是一个类别,那么y必定是相邻的类别。

再说一下建树的事。比如x,y同类,x,z是吃与被吃,那么建树时乍一看:哎呀你这吃与被吃和同类都建成父节点为x了。

事实是这样,但是,建的时候是y和z+n的父节点是x,也就是说具体的关系是通过子节点的值来判断的,不是一般并查集那样,父子节点就是同类。

#include <iostream>
#include <cstdio>

using namespace std;

const int maxn=2e6+5;
int f[maxn];

int Find(int a)
{
    if(f[a]==a) return f[a];
    else return Find(f[a]);
}

void hebing(int a,int b)
{
    int i=Find(a),j=Find(b);
    if(i==j) return;
    else f[i]=j;
}

int main()
{
    int n,m,a,b,c;
    scanf("%d%d",&n,&m);
    for(int i=0;i<=n*3;i++)
        f[i]=i;
    int sum=0;
    for(int i=0;i<m;i++)
    {
        scanf("%d%d%d",&a,&b,&c);
        if(b>n || c>n)
        {
            sum++;
            continue;
        }
        else
        {
            if(a==1)
            {
                if(Find(b)==Find(c+n) || Find(b)==Find(c+2*n))
                {
                    sum++;
                    continue;
                }
                else
                {
                    hebing(b,c);
                    hebing(b+n,c+n);
                    hebing(b+2*n,c+2*n);
                }
            }
            else
            {
                if(Find(b)==Find(c) || Find(b)==Find(c+2*n))
                {
                    sum++;
                    continue;
                }
                else
                {
                    hebing(b,c+n);
                    hebing(b+n,c+2*n);
                    hebing(b+2*n,c);
                }
            }
        }
    }
    cout << sum << endl;
    return 0;
}


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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值