C++ アルゴリズムとデータ構造のライブラリ
#include "library/polynomial/fft/subset_zeta_moebius_transform.hpp"$O(N \cdot 2^N)$
subset_zeta_transform(f);
subset_moebius_transform(f);
#pragma once
template <typename T> void subset_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 + i] += f[j + k];
}
}
}
}
template <typename T> void subset_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 + i] -= f[j + k];
}
}
}
}#line 2 "library/polynomial/fft/subset_zeta_moebius_transform.hpp"
template <typename T> void subset_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 + i] += f[j + k];
}
}
}
}
template <typename T> void subset_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 + i] -= f[j + k];
}
}
}
}