C++ アルゴリズムとデータ構造のライブラリ
#include "library/polynomial/fft/convolution_bitwise_and.hpp"$h_k = \sum_{i AND j=k} f_i \cdot g_j$
$O(NlogN)$
auto c = convolution_bitwise_and(a, b);
#pragma once
#include "library/polynomial/fft/superset_zeta_moebius_transform.hpp"
template <typename T>
vector<T> convolution_bitwise_and(vector<T> f, vector<T> g) {
const int n = (int)f.size();
assert(f.size() == g.size());
assert((n & (n - 1)) == 0);
superset_zeta_transform(f);
superset_zeta_transform(g);
for (int i = 0; i < n; i++) f[i] *= g[i];
superset_moebius_transform(f);
return f;
}#line 2 "library/polynomial/fft/superset_zeta_moebius_transform.hpp"
template <typename T> void superset_zeta_transform(vector<T> &f) {
const int n = (int)f.size();
assert((n & (n - 1)) == 0ll);
for (int i = 1; i < n; i <<= 1) {
for (int j = 0; j < n; j += i << 1) {
for (int k = 0; k < i; k++) {
f[j + k] += f[j + k + i];
}
}
}
}
template <typename T> void superset_moebius_transform(vector<T> &f) {
const int n = (int)f.size();
assert((n & (n - 1)) == 0ll);
for (int i = 1; i < n; i <<= 1) {
for (int j = 0; j < n; j += i << 1) {
for (int k = 0; k < i; k++) {
f[j + k] -= f[j + k + i];
}
}
}
}
#line 3 "library/polynomial/fft/convolution_bitwise_and.hpp"
template <typename T>
vector<T> convolution_bitwise_and(vector<T> f, vector<T> g) {
const int n = (int)f.size();
assert(f.size() == g.size());
assert((n & (n - 1)) == 0);
superset_zeta_transform(f);
superset_zeta_transform(g);
for (int i = 0; i < n; i++) f[i] *= g[i];
superset_moebius_transform(f);
return f;
}