poj 2912 Rochambeau(枚举+带权并查集)

题目链接:http://poj.org/problem?id=2912

题意:多个人玩石头剪刀布分成3组和一个裁判,每一组提前选定了自己出哪个手势,裁判可以随意出什么手势,问是否能够从给出的一系列石头剪刀布游戏中判断出哪个是裁判的,可以从第几局游戏中判断出来。

由于这题的数据量比较小可以枚举一下,枚举一下每一个人假设他们是裁判然后一系列并查集下来看

有没有出现矛盾如果没有那么这个人可能是裁判候补,如果有矛盾那么这个人肯定不是记录一下出现

问题的位置然后继续枚举。位置要去最远的,因为都要判完才知道。如果最后没有矛盾的数目大于2

说明不能确定裁判是谁,如果为0就不可能,其他则是

#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int n , m , f[510] , root[510];
struct TnT {
    int x , y;
    char cp;
}node[2010];
void init() {
    for(int i = 0 ; i < n ; i++) {
        f[i] = i , root[i] = 0;
    }
}
int find(int x) {
    if(x == f[x])
        return x;
    int tmp = find(f[x]);
    root[x] = (root[x] + root[f[x]] + 3) % 3;
    return f[x] = tmp;
}
int main() {
    int x , y;
    char s[20] , cp;
    while(scanf("%d%d" , &n , &m) != EOF) {
        for(int j = 1 ; j <= m ; j++) {
            scanf("%s" , s);
            int len = strlen(s) , temp = 0;
            x = 0 , y = 0;
            for(int i = 0 ; i < len ; i++) {
                if(s[i] == ‘=‘ || s[i] == ‘<‘ || s[i] == ‘>‘) {
                    temp = i;
                    cp = s[i];
                    break;
                }
            }
            for(int i = 0 ; i < temp ; i++) {
                x *= 10;
                x += s[i] - ‘0‘;
            }
            for(int i = temp + 1 ; i < len ; i++) {
                y *= 10;
                y += s[i] - ‘0‘;
            }
            node[j].x = x , node[j].y = y , node[j].cp = cp;
        }
        int count = 0 , pos = 0 , flag = -1 , num = 0;
        for(int i = 0 ; i < n ; i++) {
            flag = -1;
            init();
            for(int j = 1 ; j <= m ; j++) {
                if(node[j].x == i || node[j].y == i)
                    continue;
                x = node[j].x , y = node[j].y , cp = node[j].cp;
                int a = find(x) , b = find(y);
                if(a == b) {
                    if(cp == ‘=‘) {
                        if(root[x] != root[y]) {
                            flag = j;
                            break;
                        }
                    }
                    if(cp == ‘>‘) {
                        if(root[x] != (root[y] + 2) % 3) {
                            flag = j;
                            break;
                        }
                    }
                    if(cp == ‘<‘) {
                        if(root[x] != (root[y] + 1) % 3) {
                            flag = j;
                            break;
                        }
                    }
                }
                else {
                    f[a] = b;
                    if(cp == ‘=‘) {
                        root[a] = root[y] - root[x];
                        root[a] = (root[a] + 3) % 3;
                    }
                    if(cp == ‘>‘) {
                        root[a] = root[y] - root[x] + 2;
                        root[a] = (root[a] + 3) % 3;
                    }
                    if(cp == ‘<‘) {
                        root[a] = root[y] - root[x] + 1;
                        root[a] = (root[a] + 3) % 3;
                    }
                }
            }
            if(flag == -1) {
                num = i;
                count++;
            }
            else
                pos = max(pos , flag);
        }
        if(count == 0) printf("Impossible\n");
        else if(count >= 2) printf("Can not determine\n");
        else
            printf("Player %d can be determined to be the judge after %d lines\n" , num , pos);
    }
    return 0;

}
时间: 2024-10-05 17:52:24

poj 2912 Rochambeau(枚举+带权并查集)的相关文章

poj 2912 Rochambeau(带权并查集 + 暴力)

题目:poj 2912 Rochambeau(带权并查集 + 暴力) 题目大意:题目给出三个团队和一个裁判,这三个团队和裁判一起玩剪刀石头布,然后规定每个团队必须出一样的,只有裁判可以任意出.然后给出关系,x > y 代表 x 赢y , x < y代表 y 赢 x , 相等则出的一样.问这样的关系可以推出裁判是哪个吗?可以需要说明从第一条到第几条推出来的,不可以也要说明是不可能出现这样的关系,还是裁判不唯一. 解题思路:这题重点是裁判在里面会扰乱关系,并且n * m 才 100000,完全可以

POJ 1988 Cube Stacking (带权并查集)

题目链接:http://poj.org/problem?id=1988 有n个元素,开始每个元素自己 一栈,有两种操作,将含有元素x的栈放在含有y的栈的顶端,合并为一个栈.第二种操作是询问含有x元素下面有多少个元素. 经典的带权并查集,cnt表示包含这个元素的集合中所有元素个数,dis表示这个元素离最上面元素的个数(距离). 看代码领会一下吧. 1 #include <iostream> 2 #include <cstring> 3 #include <cstdio>

POJ 1182 食物链(带权并查集)

http://poj.org/problem?id=1182 题意: 动物王国中有三类动物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句话

POJ 1773 Parity game 带权并查集

分析:带权并查集,就是维护一堆关系 然后就是带权并查集的三步 1:首先确定权值数组,sum[i]代表父节点到子节点之间的1的个数(当然路径压缩后代表到根节点的个数) 1代表是奇数个,0代表偶数个 2:设计路径压缩算法 sum[x]=(sum[x]+sum[t])%2; 3:弄清合并根节点时的操作,小的在上: 注:这个题需要离散化 #include <stdio.h> #include <string.h> #include <algorithm> using names

BZOJ 3362 POJ 1984 Navigation Nightmare 带权并查集

题目大意:一些农场由一些东西向或者南北向的路相互连接.在不断加边的过程中会询问两个农场的曼哈顿距离是多少,如果目前还不连通,那么输出-1. 思路:带权并查集,f[i]为点i到father[i]的距离,要维护两个值,一个是东西向的距离,一个是南北向的距离,因为以后更新的时候要用到.在合并的时候有些特殊.现在有一条边(x->y),设fx为x的根,fy为y的根,那么现在知道f到fx的距离,y到fy的距离,还知道x到y的距离,设fx到fy的距离为dis,则dis + f[y] = f[x] + edge

POJ 1182 食物链 (带权并查集)

动物王国中有三类动物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句话有的是真的,有的是假的.当一句话满足下列三条之一时,这句话就是假话,否则

POJ 1984 - Navigation Nightmare - [带权并查集]

题目链接:http://poj.org/problem?id=1984 Time Limit: 2000MS Memory Limit: 30000K Case Time Limit: 1000MS Description Farmer John's pastoral neighborhood has N farms (2 <= N <= 40,000), usually numbered/labeled 1..N. A series of M (1 <= M < 40,000)

POJ - 1733 Parity game 带权并查集+离散化

题目大意:有10E位数,每位上的数不是1就是0.现在给出第n位到第m位的1的数量的奇偶性,判断所给出的话有几句是对的 解题思路:有10E位数,肯定要离散化处理 因为有可能给出的区间是左端点等于右端点的,所以每次都把左端点的值减去1再进行处理 给出了区间的奇偶性,就要找一下是否和前面的矛盾,如果该区间的左端点的根节点和右端点的根节点是相同的,那么就可以判断是否正确了 如果根节点不同,就进行合并,合并时要注意判断是谁的根节点比较大 #include<cstdio> #include<algo

并查集练习2(带权并查集)

明天旅游去爬山逛庙玩,今天练一天然后早早睡觉啦~ poj1703 Find them, Catch them (带权并查集) 1 #include<cstdio> 2 const int N=1e5+1; 3 int f[N]; 4 int r[N];//表示与父节点的关系,0同类,1不同类 5 int n; 6 void init(){ 7 for(int i=1;i<=n;++i){ 8 f[i]=i; r[i]=0; 9 } 10 } 11 int fin(int x){ 12 i