C++ 并查集 例题讲解

一、基本性质 
1.在基于并查集的基础上增加记录数组_rank,表示i与其同集合根节点的关系

二、例题讲解 
2.1【食物链 POJ - 1182】——vjudge题目链接

题意:
动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形。A吃B, B吃C,C吃A。 
现有N个动物,以1-N编号。每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种。 
有人用两种说法对这N个动物所构成的食物链关系进行描述: 
第一种说法是”1 X Y”,表示X和Y是同类。 
第二种说法是”2 X Y”,表示X吃Y。 
此人对N个动物,用上述两种说法,一句接一句地说出K句话,这K句话有的是真的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。 
1) 当前的话与前面的某些真的话冲突,就是假话; 
2) 当前的话中X或Y比N大,就是假话; 
3) 当前的话表示X吃X,就是假话。 
你的任务是根据给定的N(1 <= N <= 50,000)和K句话(0 <= K <= 100,000),输出假话的总数。

Input:
第一行是两个整数N和K,以一个空格分隔。 
以下K行每行是三个正整数 D,X,Y,两数之间用一个空格隔开,其中D表示说法的种类。 
若D=1,则表示X和Y是同类。 
若D=2,则表示X吃Y。

Output:
只有一个整数,表示假话的数目。

Sample Input:
100 7
1 101 1 
2 1 2
2 2 3 
2 3 3 
1 1 3 
2 3 1 
1 5 5

Sample Output:
3
1
题目分析: 
rank[x] = 0:表示x与f[x]为同种生物 
rank[x] = 1:表示x吃f[x] 
rank[x] = 2:表示x被f[x]吃

以下为Accepted代码
 

#include 
#include 
#include 

using namespace std;

const int N = 54014;

int f[N], _rank[N];

void init(int n);
int get_f(int x);
bool Merge(int d, int x, int y);

int main(){
    int n, m, d, x, y, cnt;
    scanf("%d %d", &n, &m);
    init(n);
    cnt = 0;
    while(m--){
        scanf("%d %d %d", &d, &x, &y);
        if(x > n || y > n){
            cnt++;
            continue;
        }
        if(d == 2 && x == y){
            cnt++;
            continue;
        }
        if(!Merge(d, x, y)){
            cnt++;
        }
    }
    printf("%d\n", cnt);
    return 0;
}
void init(int n){
    for(int i = 1; i <= n; i++){
        f[i] = i, _rank[i] = 0;
    }
}
int get_f(int x){
    if(x == f[x]) return x;
    int fa = f[x];/*记录x未直接连接到根节点之前的父亲节点*/
    f[x] = get_f(f[x]);
    ///通过枚举x与父节点的_rank记录关系得到x与父节点的父节点的_rank记录关系
    if(_rank[x] == 0 && _rank[fa] == 0) _rank[x] = 0;
    else if(_rank[x] == 0 && _rank[fa] == 1) _rank[x] = 1;
    else if(_rank[x] == 0 && _rank[fa] == 2) _rank[x] = 2;
    else if(_rank[x] == 1 && _rank[fa] == 0) _rank[x] = 1;
    else if(_rank[x] == 1 && _rank[fa] == 1) _rank[x] = 2;
    else if(_rank[x] == 1 && _rank[fa] == 2) _rank[x] = 0;
    else if(_rank[x] == 2 && _rank[fa] == 0) _rank[x] = 2;
    else if(_rank[x] == 2 && _rank[fa] == 1) _rank[x] = 0;
    else if(_rank[x] == 2 && _rank[fa] == 2) _rank[x] = 1;
    return f[x];
}
bool Merge(int d, int x, int y){
    int t1 = get_f(x);
    int t2 = get_f(y);
    if(d == 1){
        if(t1 == t2){
            if(_rank[x] != _rank[y]) return false;
            else return true;
        }
        else {
            f[t1] = t2;
            ///通过枚举x与t1的_rank记录关系和y与t2的_rank记录关系得到t2与t1的_rank记录关系
            if(_rank[x] == 0 && _rank[y] == 0) _rank[t1] = 0;
            else if(_rank[x] == 0 && _rank[y] == 1) _rank[t1] = 1;
            else if(_rank[x] == 0 && _rank[y] == 2) _rank[t1] = 2;
            else if(_rank[x] == 1 && _rank[y] == 0) _rank[t1] = 2;
            else if(_rank[x] == 1 && _rank[y] == 1) _rank[t1] = 0;
            else if(_rank[x] == 1 && _rank[y] == 2) _rank[t1] = 1;
            else if(_rank[x] == 2 && _rank[y] == 0) _rank[t1] = 1;
            else if(_rank[x] == 2 && _rank[y] == 1) _rank[t1] = 2;
            else if(_rank[x] == 2 && _rank[y] == 2) _rank[t1] = 0;
            return true;
        }
    }
    else {/*d == 2*/
        if(t1 == t2){
            if(_rank[x] == 0 && _rank[y] != 2) return false;
            else if(_rank[x] == 1 && _rank[y] != 0) return false;
            else if(_rank[x] == 2 && _rank[y] != 1) return false;
            else return true;
        }
        else {
            f[t1] = t2;
            ///通过枚举x与t1的_rank记录关系和y与t2的_rank记录关系得到t2与t1的_rank记录关系
            if(_rank[x] == 0 && _rank[y] == 0) _rank[t1] = 1;
            else if(_rank[x] == 0 && _rank[y] == 1) _rank[t1] = 2;
            else if(_rank[x] == 0 && _rank[y] == 2) _rank[t1] = 0;
            else if(_rank[x] == 1 && _rank[y] == 0) _rank[t1] = 0;
            else if(_rank[x] == 1 && _rank[y] == 1) _rank[t1] = 1;
            else if(_rank[x] == 1 && _rank[y] == 2) _rank[t1] = 2;
            else if(_rank[x] == 2 && _rank[y] == 0) _rank[t1] = 2;
            else if(_rank[x] == 2 && _rank[y] == 1) _rank[t1] = 0;
            else if(_rank[x] == 2 && _rank[y] == 2) _rank[t1] = 1;
            return true;
        }
    }
}

 

2.2【Por Costel and the Match Gym - 100923H——vjudge题目链接】

题意:
已知有n个战士,分属于两个红、蓝阵营,现输入m条敌对关系,当本条关系信息与之前的真的关系信息冲突,则认为本条关系信息为假(即先输入的敌对关系优先为真)

Input:
第一行输入T,代表T组测试数据 
每组测试数据第一行输入整数n和m,表示n个战士,m条敌对关系判断语句 
之后输入m行,每行含有整数x和y,表示认为x和y为敌对关系(即阵营不同)

Output:
输出m行,每行输出”YES”或”NO”(不含引号),表示判断所得本条语句的真假

Sample Input:
1
3 3
1 2
2 3
1 3

Sample Output:
YES
YES
NO

题目分析: 
rank[x] = 0:表示x与f[x]为朋友(即属于同一阵营) 
rank[x] = 1:表示x与f[x]为敌人(即分属不同阵营)

以下为Accepted代码
 

#include 
#include 
#include 

using namespace std;

const int N = 104014;

int f[N], _rank[N];

void init(int n);
int get_f(int x);
bool Merge(int x, int y);

int main(){
    freopen("meciul.in", "r", stdin);
    freopen("meciul.out", "w", stdout);
    int T, n, m, x, y;
    scanf("%d", &T);
    while(T--){
        scanf("%d %d", &n, &m);
        init(n);
        while(m--){
            scanf("%d %d", &x, &y);
            if(Merge(x, y)) printf("YES\n");
            else printf("NO\n");
        }
    }
    return 0;
}
void init(int n){
    for(int i = 1; i <= n; i++){
        f[i] = i, _rank[i] = 0;/*初始化:各自为根、皆为朋友*/
    }
}
int get_f(int x){
    if(x == f[x]) return x;
    int fa = f[x];/*记录未路径压缩(直接与根节点相连)前的父亲节点*/
    f[x] = get_f(f[x]);
    if(_rank[x] == 0 && _rank[fa] == 0) _rank[x] = 0;
    else if(_rank[x] == 0 && _rank[fa] == 1) _rank[x] = 1;
    else if(_rank[x] == 1 && _rank[fa] == 0) _rank[x] = 1;
    else if(_rank[x] == 1 && _rank[fa] == 1) _rank[x] = 0;
    return f[x];
}
bool Merge(int x, int y){
    int t1 = get_f(x);
    int t2 = get_f(y);
    if(t1 == t2){/*若x与y已存在关系,则通过x与根节点的_rank记录关系、y与根节点的_rank记录关系判断x与y是否为敌对关系*/
        if(_rank[x] == 0 && _rank[y] == 0) return false;
        if(_rank[x] == 0 && _rank[y] == 1) return true;
        if(_rank[x] == 1 && _rank[y] == 0) return true;
        if(_rank[x] == 1 && _rank[y] == 1) return false;
    }
    else {/*若x与y尚未存在关系,则根据x与其根节点t1的关系、y与其根节点t2的关系连接两个关系集合(连接两个关系集合的根)*/
        f[t1] = t2;
        if(_rank[x] == 0 && _rank[y] == 0) _rank[t1] = 1;
        else if(_rank[x] == 0 && _rank[y] == 1) _rank[t1] = 0;
        else if(_rank[x] == 1 && _rank[y] == 0) _rank[t1] = 0;
        else if(_rank[x] == 1 && _rank[y] == 1) _rank[t1] = 1;
    }
    return true;
}

2.2例题感悟:1.基于并查集;2.认真仔细寻找rank[i]表示含义

你可能感兴趣的:(c++)