Post

【精進8】ZONe 2021-F

【精進8】ZONe 2021-F

XOR基底シリーズ

問題

$N=2^n$頂点がある。長さ$M$の列$A$が与えられ、任意の$i$について$s\oplus t \ne A_i$のとき頂点$s,t$間に辺を張ることができる。連結にできるか判定し、できるなら$1$つ辺の貼り方を出力せよ。

解法

$A$よりも$A$にない数に注目する。$B$を$0$以上$N$未満の数であって$A$のどの要素とも異なるものの集合とする。$B$の張る空間 $X(B)=\lbrace\bigoplus_{s\in S}s \mid{S\subseteq B}\rbrace$ の基底であって、$B$の部分集合となるものの$1$つを$\mathrm{basis}(B)$と書く。 このとき頂点$0$と連結にできる集合は$X(B)$である。実際$x\in X(B)$について次のように辺を構成できる。まずある$E \subseteq \mathrm{basis}(B)\subseteq B$について$x=\bigoplus_{e\in E}e$と書ける。$E=\lbrace e_1,\dots,e_K\rbrace,e_0=0$と置き、$t_i=\bigoplus_{j\le i}e_j$とすると、各$i$について$t_i\oplus t_{i+1}=e_{i+1}\in B$であるから頂点$t_i,t_{i+1}$間に辺を貼れる。$t_0=0,t_K=x$であるから$0$と$x$を連結となるように辺を張ることができた。また任意の$s\not\in X(B),x\in X(B)$について$s\oplus x\not\in B$となるから$0$と$s\not \in X(B)$は連結にできない。従って$|X(B)|=N$が連結にできる条件である。辺の構築については、順に$X,\mathrm{basis}$を構築する中で、$\mathrm{basis}$が増えた際に元の$X$の要素$x$と$e_{\mathrm{new}}\oplus x$間に辺を張っていけばよい。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
#include "template.hpp"
void solve() {
    LL(n, m);
    vl a(n, 0);
    rep(i, m) {
        LL(ai);
        a[ai]++;
    }
    vl basis;
    vector<P> ans;
    vl x = {0}; // 0と連結な頂点の集合
    auto in = [&](const ll num) {
        ll e_new = num;
        for(auto &e : basis)
            chmin(e_new, e_new ^ e);
        if(e_new == 0)
            return;
        basis.push_back(e_new);
        ll sz = x.size();
        rep(i, sz) {
            ans.push_back({x[i], x[i] ^ num});
            x.push_back(x[i] ^ num);
        }
    };
    rep(i, n) {
        if(a[i] == 0)
            in(i);
    }
    if(x.size() < n) {
        print(-1);
    } else {
        for(auto &[i, j] : ans) {
            print(i, j);
        }
    }
}

int main() {
    ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    cout << std::setprecision(16);
    int t = 1;
    rep(_, t) {
        solve();
    }
}

This post is licensed under CC BY 4.0 by the author.