正经写了一个带权并查集的简单题。

这个题的操作一共有两种:第一种是告诉你\(X_p\ \text{xor}\ X_q = v\),要判断与之前的信息是否矛盾。第二种是让你求一系列\(X\)的xor和,同时也要判断不知道的情况。

不要问我为什么题目中是三个操作,因为第一个操作就是第二个操作。

首先你需要读入,这个可以用sscanf。然后就可以开始带权并查集搞了。

首先并查集里维护\(xor_{u}\),代表\(X_u\ \text{xor}\ X_{f_u}\)的值。这个东西一开始都不知道,所以都设成0.

当进行路径压缩的时候这么写:

int find(int x) {
    if (x == f[x]) return x;
    int get = find(f[x]);
    xor_[x] ^= xor_[f[x]];
    return f[x] = get;
}

这个东西成立是因为\((a\ xor\ b)\ xor\ (b\ xor\ c)=a\ xor\ c\).

然后如果他告诉你了一条关系,你先找到对应并查集的代表,然后合并一下,注意要特判下恒0的情况。

这里合并要判断他俩在不在一个集合里,可以结合代码理解:

bool merge(int p, int q, int v) {
    int fp = find(p), fq = find(q);
    if (fp == fq) {
        if ((xor_[p] ^ xor_[q]) != v) return 0;
        return 1;
    }
    if (fp == n) swap(fp, fq);
    f[fp] = fq;
    xor_[fp] = (xor_[p] ^ xor_[q] ^ v);
    return 1;
}

求那个东西也挺有意思的。

考虑把在一个森林里的所有东西一块计算。

注意到我们能搞出来的xor和=x1 xor xf xor x2 xor xf xor...xor xp xor xf

如果我们消不掉xf,除非xf=0,否则一定无解。

否则直接消即可。不知道为啥uva一直过不了,但是hdu就能过...

#include <bits/stdc++.h>

#define test(...) fprintf(stderr, __VA_ARGS__)
#define dbg(x) cerr << #x << " = " << x << '\n'

using namespace std;

typedef long long ll;
typedef pair <int, int> pii;
typedef vector <int> vi;
typedef unsigned int ui;
typedef vector <pair <int, int> > edges;

const int N = 1000005;
char s[N], from[N];
int f[N], xor_[N], n, q;
// xor_[x]: x -> fa[x]这条边的xor值
// 路径压缩的时候顺带维护
int find(int x) {
    if (x == f[x]) return x;
    int get = find(f[x]);
    xor_[x] ^= xor_[f[x]];
    return f[x] = get;
}
int case_num;
bool merge(int p, int q, int v) {
    int fp = find(p), fq = find(q);
    if (fp == fq) {
        if ((xor_[p] ^ xor_[q]) != v) return 0;
        return 1;
    }
    if (fp == n) swap(fp, fq);
    f[fp] = fq;
    xor_[fp] = (xor_[p] ^ xor_[q] ^ v);
    return 1;
}

int ans(vi now) {
    int k = now.size(), ans = 0;
    vi ind(k);
    for (int i = 0; i < k; ++i) {
        if (ind[i]) continue;

        int fi = find(now[i]), cnt = 0;
        for (int j = i; j < k; ++j) {
            int fj = find(now[j]);
            if (!ind[j] && fi == fj) {
                cnt++;
                ind[j] = 1;
                ans ^= xor_[now[j]];
            }
        }
        if ((cnt & 1) && fi != n) return -1;
    }
    return ans;
}
void solve() {
    printf("Case %d:\n", ++case_num);
    for (int i = 0; i <= 1000000; ++i)
        f[i] = i, xor_[i] = 0;
    bool flag = 0;
    int facts = 0;
    while (q--) {
        char op[3];
        scanf("%s", op);
        if (op[0] == 'I') {
            gets(from);
            int space = 0, p, q, v;
            for (int i = 1; from[i] != '\0'; ++i)
                if (from[i] == ' ') ++space;
    //      dbg(space);
            if (space == 1) {
                sscanf(from, "%d%d", &p, &v);
                q = n;
            } else if (space == 2)
                sscanf(from, "%d%d%d", &p, &q, &v);
            ++facts;
            if (flag) continue;
            if (!merge(p, q, v)) {
                printf("The first %d facts are conflicting.\n", facts);
                flag = 1;
            }
        } else if (op[0] == 'Q') {
            int k;
            scanf ("%d", &k);
            vi q(k);
            for (int i = 0; i < k; ++i) scanf ("%d", &q[i]);

            if (flag) continue;
            int result = ans(q);
            if (result == -1) {
                puts("I don't know.");
            }
            else printf("%d\n", result);
        }
    }
    puts("");
}

int main() {
#ifdef LOCAL
    freopen("sample.in", "r", stdin);
#endif
  while (~scanf("%d%d", &n, &q) && (n || q)) solve();
  return 0;
}
02-14 03:23