[2019.1.1]BZOJ4195 [Noi2015]程序自动分析

首先很容易想到并查集维护,将相等的数merge起来。但是我们很难维护不等的情况。

那怎么办?

我们发现我们可以查询两数不等是否成立,只是不能维护它而已,并且事实上,不等没有类似\(a\ne b,b\ne c,\texttt{则}a\ne c\)的性质。

所以我们可以变更维护顺序。

先在并查集里维护等于的关系(也就是先处理\(e=1\)的约束),然后对于每一个不等约束条件\(a\ne b\),判断\(a,b\)是否在同一并查集里。

至于数据范围是\(10^9\),开个map把每一个出现的未知数编号对应到\([1,n]\)范围内的整数上就好了。

code:

#include<bits/stdc++.h>
using namespace std;
int T,n,u,v,cnt,tu,tv,op,ne[100010][2],sz,f[200010];
map<int,int>mp;
int getf(int x){
    return f[x]==x?x:f[x]=getf(f[x]);
}
void merge(int x,int y){
    if(getf(x)==getf(y))return;
    f[f[x]]=f[y];
}
int main(){
    scanf("%d",&T);
    while(T--){
        scanf("%d",&n);
        for(int i=1;i<=2*n;i++)f[i]=i;
        mp.clear();
        cnt=0;
        sz=0;
        for(int i=1;i<=n;i++){
            scanf("%d%d%d",&u,&v,&op);
            tu=mp[u];
            if(!tu)tu=mp[u]=++cnt;
            tv=mp[v];
            if(!tv)tv=mp[v]=++cnt;
            if(!op)ne[++sz][0]=tv,ne[sz][1]=tu;
            else merge(tu,tv);
        }
        for(int i=1;i<=cnt;i++)getf(i);
        for(int i=1;i<=sz;i++)
            if(f[ne[i][0]]==f[ne[i][1]]){
                puts("NO");
                goto END;
            }
        puts("YES");
        END:
        ;
    }
    return 0;
}
posted @ 2019-03-17 17:54  xryjr233  阅读(130)  评论(0编辑  收藏  举报