serna37's Library

Logo

C++ アルゴリズムとデータ構造のライブラリ

View the Project on GitHub serna37/library-cpp

:heavy_check_mark: Bitwise畳み込みXOR
(library/polynomial/fft/convolution_bitwise_xor.hpp)

Bitwise畳み込みXOR

詳しい説明

できること

$h_k = \sum_{i \oplus j=k} f_i \cdot g_j$

計算量

$O(NlogN)$

使い方

auto c = convolution_bitwise_xor(a, b);

Depends on

Verified with

Code

#pragma once
#include "library/polynomial/fft/fast_walsh_hadamard_transform.hpp"
template <typename T>
vector<T> convolution_bitwise_xor(vector<T> f, vector<T> g) {
    const int n = (int)f.size();
    assert(f.size() == g.size());
    assert((n & (n - 1)) == 0);
    fast_walsh_hadamard_transform(f);
    fast_walsh_hadamard_transform(g);
    for (int i = 0; i < n; i++) f[i] *= g[i];
    fast_walsh_hadamard_transform(f, true);
    return f;
}
#line 2 "library/polynomial/fft/fast_walsh_hadamard_transform.hpp"
template <typename T>
void fast_walsh_hadamard_transform(vector<T> &f, bool inv = false) {
    const int n = (int)f.size();
    assert((n & (n - 1)) == 0);
    for (int i = 1; i < n; i <<= 1) {
        for (int j = 0; j < n; j += i << 1) {
            for (int k = 0; k < i; k++) {
                T s = f[j + k], t = f[j + k + i];
                f[j + k] = s + t;
                f[j + k + i] = s - t;
            }
        }
    }
    if (inv) {
        T inv_n = T(1) / n;
        for (auto &x : f) x *= inv_n;
    }
}
#line 3 "library/polynomial/fft/convolution_bitwise_xor.hpp"
template <typename T>
vector<T> convolution_bitwise_xor(vector<T> f, vector<T> g) {
    const int n = (int)f.size();
    assert(f.size() == g.size());
    assert((n & (n - 1)) == 0);
    fast_walsh_hadamard_transform(f);
    fast_walsh_hadamard_transform(g);
    for (int i = 0; i < n; i++) f[i] *= g[i];
    fast_walsh_hadamard_transform(f, true);
    return f;
}
Back to top page