This documentation is automatically generated by online-judge-tools/verification-helper
#define PROBLEM "https://judge.yosupo.jp/problem/stirling_number_of_the_first_kind_fixed_k"
#include "famous_sequence.hpp"
#include "modint.hpp"
#include <iostream>
#include <vector>
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
using mint = ModInt<998244353>;
int n, k;
std::cin >> n >> k;
const auto S = signed_stirling_numbers_1st_column<mint>(k, n + 1);
for (int i = k; i <= n; ++i) std::cout << S[i] << ' ';
return 0;
}
#line 1 "test/enumerative_combinatorics/stirling_number_of_the_first_kind_fixed_k.0.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/stirling_number_of_the_first_kind_fixed_k"
#line 2 "famous_sequence.hpp"
#line 2 "fft.hpp"
#include <algorithm>
#include <cassert>
#include <iterator>
#include <memory>
#include <vector>
template<typename Tp> class FftInfo {
static Tp least_quadratic_nonresidue() {
for (int i = 2;; ++i)
if (Tp(i).pow((Tp::mod() - 1) / 2) == -1) return Tp(i);
}
const int ordlog2_;
const Tp zeta_;
const Tp invzeta_;
const Tp imag_;
const Tp invimag_;
mutable std::vector<Tp> root_;
mutable std::vector<Tp> invroot_;
FftInfo()
: ordlog2_(__builtin_ctzll(Tp::mod() - 1)),
zeta_(least_quadratic_nonresidue().pow((Tp::mod() - 1) >> ordlog2_)),
invzeta_(zeta_.inv()), imag_(zeta_.pow(1LL << (ordlog2_ - 2))), invimag_(-imag_),
root_{Tp(1), imag_}, invroot_{Tp(1), invimag_} {}
public:
static const FftInfo &get() {
static FftInfo info;
return info;
}
Tp imag() const { return imag_; }
Tp inv_imag() const { return invimag_; }
Tp zeta() const { return zeta_; }
Tp inv_zeta() const { return invzeta_; }
const std::vector<Tp> &root(int n) const {
// [0, n)
assert((n & (n - 1)) == 0);
if (const int s = root_.size(); s < n) {
root_.resize(n);
for (int i = __builtin_ctz(s); (1 << i) < n; ++i) {
const int j = 1 << i;
root_[j] = zeta_.pow(1LL << (ordlog2_ - i - 2));
for (int k = j + 1; k < j * 2; ++k) root_[k] = root_[k - j] * root_[j];
}
}
return root_;
}
const std::vector<Tp> &inv_root(int n) const {
// [0, n)
assert((n & (n - 1)) == 0);
if (const int s = invroot_.size(); s < n) {
invroot_.resize(n);
for (int i = __builtin_ctz(s); (1 << i) < n; ++i) {
const int j = 1 << i;
invroot_[j] = invzeta_.pow(1LL << (ordlog2_ - i - 2));
for (int k = j + 1; k < j * 2; ++k) invroot_[k] = invroot_[k - j] * invroot_[j];
}
}
return invroot_;
}
};
inline int fft_len(int n) {
--n;
n |= n >> 1, n |= n >> 2, n |= n >> 4, n |= n >> 8;
return (n | n >> 16) + 1;
}
namespace detail {
template<typename Iterator> inline void
butterfly_n(Iterator a, int n,
const std::vector<typename std::iterator_traits<Iterator>::value_type> &root) {
assert(n > 0);
assert((n & (n - 1)) == 0);
const int bn = __builtin_ctz(n);
if (bn & 1) {
for (int i = 0; i < n / 2; ++i) {
const auto a0 = a[i], a1 = a[i + n / 2];
a[i] = a0 + a1, a[i + n / 2] = a0 - a1;
}
}
for (int i = n >> (bn & 1); i >= 4; i /= 4) {
const int i4 = i / 4;
for (int k = 0; k < i4; ++k) {
const auto a0 = a[k + i4 * 0], a1 = a[k + i4 * 1];
const auto a2 = a[k + i4 * 2], a3 = a[k + i4 * 3];
const auto a02p = a0 + a2, a02m = a0 - a2;
const auto a13p = a1 + a3, a13m = (a1 - a3) * root[1];
a[k + i4 * 0] = a02p + a13p, a[k + i4 * 1] = a02p - a13p;
a[k + i4 * 2] = a02m + a13m, a[k + i4 * 3] = a02m - a13m;
}
for (int j = i, m = 2; j < n; j += i, m += 2) {
const auto r = root[m], r2 = r * r, r3 = r2 * r;
for (int k = j; k < j + i4; ++k) {
const auto a0 = a[k + i4 * 0], a1 = a[k + i4 * 1] * r;
const auto a2 = a[k + i4 * 2] * r2, a3 = a[k + i4 * 3] * r3;
const auto a02p = a0 + a2, a02m = a0 - a2;
const auto a13p = a1 + a3, a13m = (a1 - a3) * root[1];
a[k + i4 * 0] = a02p + a13p, a[k + i4 * 1] = a02p - a13p;
a[k + i4 * 2] = a02m + a13m, a[k + i4 * 3] = a02m - a13m;
}
}
}
}
template<typename Iterator> inline void
inv_butterfly_n(Iterator a, int n,
const std::vector<typename std::iterator_traits<Iterator>::value_type> &root) {
assert(n > 0);
assert((n & (n - 1)) == 0);
const int bn = __builtin_ctz(n);
for (int i = 4; i <= (n >> (bn & 1)); i *= 4) {
const int i4 = i / 4;
for (int k = 0; k < i4; ++k) {
const auto a0 = a[k + i4 * 0], a1 = a[k + i4 * 1];
const auto a2 = a[k + i4 * 2], a3 = a[k + i4 * 3];
const auto a01p = a0 + a1, a01m = a0 - a1;
const auto a23p = a2 + a3, a23m = (a2 - a3) * root[1];
a[k + i4 * 0] = a01p + a23p, a[k + i4 * 1] = a01m + a23m;
a[k + i4 * 2] = a01p - a23p, a[k + i4 * 3] = a01m - a23m;
}
for (int j = i, m = 2; j < n; j += i, m += 2) {
const auto r = root[m], r2 = r * r, r3 = r2 * r;
for (int k = j; k < j + i4; ++k) {
const auto a0 = a[k + i4 * 0], a1 = a[k + i4 * 1];
const auto a2 = a[k + i4 * 2], a3 = a[k + i4 * 3];
const auto a01p = a0 + a1, a01m = a0 - a1;
const auto a23p = a2 + a3, a23m = (a2 - a3) * root[1];
a[k + i4 * 0] = a01p + a23p, a[k + i4 * 1] = (a01m + a23m) * r;
a[k + i4 * 2] = (a01p - a23p) * r2, a[k + i4 * 3] = (a01m - a23m) * r3;
}
}
}
if (bn & 1) {
for (int i = 0; i < n / 2; ++i) {
const auto a0 = a[i], a1 = a[i + n / 2];
a[i] = a0 + a1, a[i + n / 2] = a0 - a1;
}
}
}
} // namespace detail
// FFT_n: A(x) |-> bit-reversed order of [A(1), A(zeta_n), ..., A(zeta_n^(n-1))]
template<typename Iterator> inline void fft_n(Iterator a, int n) {
using Tp = typename std::iterator_traits<Iterator>::value_type;
detail::butterfly_n(a, n, FftInfo<Tp>::get().root(n / 2));
}
template<typename Tp> inline void fft(std::vector<Tp> &a) { fft_n(a.begin(), a.size()); }
// IFFT_n: bit-reversed order of [A(1), A(zeta_n), ..., A(zeta_n^(n-1))] |-> A(x)
template<typename Iterator> inline void inv_fft_n(Iterator a, int n) {
using Tp = typename std::iterator_traits<Iterator>::value_type;
detail::inv_butterfly_n(a, n, FftInfo<Tp>::get().inv_root(n / 2));
const Tp iv = Tp::mod() - (Tp::mod() - 1) / n;
for (int i = 0; i < n; ++i) a[i] *= iv;
}
template<typename Tp> inline void inv_fft(std::vector<Tp> &a) { inv_fft_n(a.begin(), a.size()); }
// IFFT_n^T: A(x) |-> 1/n FFT_n((x^n A(x^(-1))) mod (x^n - 1))
template<typename Iterator> inline void transposed_inv_fft_n(Iterator a, int n) {
using Tp = typename std::iterator_traits<Iterator>::value_type;
const Tp iv = Tp::mod() - (Tp::mod() - 1) / n;
for (int i = 0; i < n; ++i) a[i] *= iv;
detail::butterfly_n(a, n, FftInfo<Tp>::get().inv_root(n / 2));
}
template<typename Tp> inline void transposed_inv_fft(std::vector<Tp> &a) {
transposed_inv_fft_n(a.begin(), a.size());
}
// FFT_n^T : FFT_n((x^n A(x^(-1))) mod (x^n - 1)) |-> n A(x)
template<typename Iterator> inline void transposed_fft_n(Iterator a, int n) {
using Tp = typename std::iterator_traits<Iterator>::value_type;
detail::inv_butterfly_n(a, n, FftInfo<Tp>::get().root(n / 2));
}
template<typename Tp> inline void transposed_fft(std::vector<Tp> &a) {
transposed_fft_n(a.begin(), a.size());
}
template<typename Tp> inline std::vector<Tp> convolution_fft(std::vector<Tp> a, std::vector<Tp> b) {
if (a.empty() || b.empty()) return {};
const int n = a.size();
const int m = b.size();
const int len = fft_len(n + m - 1);
a.resize(len);
b.resize(len);
fft(a);
fft(b);
for (int i = 0; i < len; ++i) a[i] *= b[i];
inv_fft(a);
a.resize(n + m - 1);
return a;
}
template<typename Tp> inline std::vector<Tp> square_fft(std::vector<Tp> a) {
if (a.empty()) return {};
const int n = a.size();
const int len = fft_len(n * 2 - 1);
a.resize(len);
fft(a);
for (int i = 0; i < len; ++i) a[i] *= a[i];
inv_fft(a);
a.resize(n * 2 - 1);
return a;
}
template<typename Tp>
inline std::vector<Tp> convolution_naive(const std::vector<Tp> &a, const std::vector<Tp> &b) {
if (a.empty() || b.empty()) return {};
const int n = a.size();
const int m = b.size();
std::vector<Tp> res(n + m - 1);
for (int i = 0; i < n; ++i)
for (int j = 0; j < m; ++j) res[i + j] += a[i] * b[j];
return res;
}
template<typename Tp>
inline std::vector<Tp> convolution(const std::vector<Tp> &a, const std::vector<Tp> &b) {
if (std::min(a.size(), b.size()) < 60) return convolution_naive(a, b);
if (std::addressof(a) == std::addressof(b)) return square_fft(a);
return convolution_fft(a, b);
}
#line 2 "fps_basic.hpp"
#line 2 "binomial.hpp"
#line 5 "binomial.hpp"
template<typename Tp> class Binomial {
std::vector<Tp> factorial_, invfactorial_;
Binomial() : factorial_{Tp(1)}, invfactorial_{Tp(1)} {}
void preprocess(int n) {
if (const int nn = factorial_.size(); nn < n) {
int k = nn;
while (k < n) k *= 2;
k = std::min<long long>(k, Tp::mod());
factorial_.resize(k);
invfactorial_.resize(k);
for (int i = nn; i < k; ++i) factorial_[i] = factorial_[i - 1] * i;
invfactorial_.back() = factorial_.back().inv();
for (int i = k - 2; i >= nn; --i) invfactorial_[i] = invfactorial_[i + 1] * (i + 1);
}
}
public:
static const Binomial &get(int n) {
static Binomial bin;
bin.preprocess(n);
return bin;
}
Tp binom(int n, int m) const {
return n < m ? Tp() : factorial_[n] * invfactorial_[m] * invfactorial_[n - m];
}
Tp inv(int n) const { return factorial_[n - 1] * invfactorial_[n]; }
Tp factorial(int n) const { return factorial_[n]; }
Tp inv_factorial(int n) const { return invfactorial_[n]; }
};
#line 2 "semi_relaxed_conv.hpp"
#line 5 "semi_relaxed_conv.hpp"
#include <type_traits>
#include <utility>
#line 8 "semi_relaxed_conv.hpp"
template<typename Tp, typename Closure>
inline std::enable_if_t<std::is_invocable_r_v<Tp, Closure, int, const std::vector<Tp> &>,
std::vector<Tp>>
semi_relaxed_convolution_naive(const std::vector<Tp> &A, Closure gen, int n) {
std::vector<Tp> B(n), AB(n);
for (int i = 0; i < n; ++i) {
for (int j = std::max(0, i - (int)A.size() + 1); j < i; ++j) AB[i] += A[i - j] * B[j];
B[i] = gen(i, AB);
if (!A.empty()) AB[i] += A[0] * B[i];
}
return B;
}
// returns coefficients generated by closure
// closure: gen(index, current_product)
template<typename Tp, typename Closure>
inline std::enable_if_t<std::is_invocable_r_v<Tp, Closure, int, const std::vector<Tp> &>,
std::vector<Tp>>
semi_relaxed_convolution(const std::vector<Tp> &A, Closure gen, int n) {
if (A.size() < 60) return semi_relaxed_convolution_naive(A, gen, n);
enum { BaseCaseSize = 32 };
static_assert((BaseCaseSize & (BaseCaseSize - 1)) == 0);
static const int Block[] = {16, 16, 16, 16, 16};
static const int BlockSize[] = {
BaseCaseSize,
BaseCaseSize * Block[0],
BaseCaseSize * Block[0] * Block[1],
BaseCaseSize * Block[0] * Block[1] * Block[2],
BaseCaseSize * Block[0] * Block[1] * Block[2] * Block[3],
BaseCaseSize * Block[0] * Block[1] * Block[2] * Block[3] * Block[4],
};
// returns (which_block, level)
auto blockinfo = [](int ind) {
int i = ind / BaseCaseSize, lv = 0;
while ((i & (Block[lv] - 1)) == 0) i /= Block[lv++];
return std::make_pair(i & (Block[lv] - 1), lv);
};
std::vector<Tp> B(n), AB(n);
std::vector<std::vector<std::vector<Tp>>> dftA, dftB;
for (int i = 0; i < n; ++i) {
const int s = i & (BaseCaseSize - 1);
// block contribution
if (i >= BaseCaseSize && s == 0) {
const auto [j, lv] = blockinfo(i);
const int blocksize = BlockSize[lv];
if (blocksize * j == i) {
if ((int)dftA.size() == lv) {
dftA.emplace_back();
dftB.emplace_back(Block[lv] - 1);
}
if ((j - 1) * blocksize < (int)A.size()) {
dftA[lv]
.emplace_back(A.begin() + (j - 1) * blocksize,
A.begin() + std::min<int>((j + 1) * blocksize, A.size()))
.resize(blocksize * 2);
fft(dftA[lv][j - 1]);
}
}
if (!dftA[lv].empty()) {
dftB[lv][j - 1].resize(blocksize * 2);
std::copy_n(B.begin() + (i - blocksize), blocksize, dftB[lv][j - 1].begin());
std::fill_n(dftB[lv][j - 1].begin() + blocksize, blocksize, Tp(0));
fft(dftB[lv][j - 1]);
// middle product
std::vector<Tp> mp(blocksize * 2);
for (int k = 0; k < std::min<int>(j, dftA[lv].size()); ++k)
for (int l = 0; l < blocksize * 2; ++l)
mp[l] += dftA[lv][k][l] * dftB[lv][j - 1 - k][l];
inv_fft(mp);
for (int k = 0; k < blocksize && i + k < n; ++k) AB[i + k] += mp[k + blocksize];
}
}
// basecase contribution
for (int j = std::max(i - s, i - (int)A.size() + 1); j < i; ++j) AB[i] += A[i - j] * B[j];
B[i] = gen(i, AB);
if (!A.empty()) AB[i] += A[0] * B[i];
}
return B;
}
#line 8 "fps_basic.hpp"
template<typename Tp> inline int order(const std::vector<Tp> &a) {
for (int i = 0; i < (int)a.size(); ++i)
if (a[i] != 0) return i;
return -1;
}
template<typename Tp> inline std::vector<Tp> fps_inv(const std::vector<Tp> &a, int n) {
assert(order(a) == 0);
if (n <= 0) return {};
if (std::min<int>(a.size(), n) < 60)
return semi_relaxed_convolution(
a, [v = a[0].inv()](int n, auto &&c) { return n == 0 ? v : -c[n] * v; }, n);
enum { Threshold = 32 };
const int len = fft_len(n);
std::vector<Tp> invA, shopA(len), shopB(len);
invA = semi_relaxed_convolution(
a, [v = a[0].inv()](int n, auto &&c) { return n == 0 ? v : -c[n] * v; }, Threshold);
invA.resize(len);
for (int i = Threshold * 2; i <= len; i *= 2) {
std::fill(std::copy_n(a.begin(), std::min<int>(a.size(), i), shopA.begin()),
shopA.begin() + i, Tp(0));
std::copy_n(invA.begin(), i, shopB.begin());
fft_n(shopA.begin(), i);
fft_n(shopB.begin(), i);
for (int j = 0; j < i; ++j) shopA[j] *= shopB[j];
inv_fft_n(shopA.begin(), i);
std::fill_n(shopA.begin(), i / 2, Tp(0));
fft_n(shopA.begin(), i);
for (int j = 0; j < i; ++j) shopA[j] *= shopB[j];
inv_fft_n(shopA.begin(), i);
for (int j = i / 2; j < i; ++j) invA[j] = -shopA[j];
}
invA.resize(n);
return invA;
}
template<typename Tp>
inline std::vector<Tp> fps_div(const std::vector<Tp> &a, const std::vector<Tp> &b, int n) {
assert(order(b) == 0);
if (n <= 0) return {};
return semi_relaxed_convolution(
b,
[&, v = b[0].inv()](int n, auto &&c) {
if (n < (int)a.size()) return (a[n] - c[n]) * v;
return -c[n] * v;
},
n);
}
template<typename Tp> inline std::vector<Tp> deriv(const std::vector<Tp> &a) {
const int n = (int)a.size() - 1;
if (n <= 0) return {};
std::vector<Tp> res(n);
for (int i = 1; i <= n; ++i) res[i - 1] = a[i] * i;
return res;
}
template<typename Tp> inline std::vector<Tp> integr(const std::vector<Tp> &a, Tp c = {}) {
const int n = a.size() + 1;
auto &&bin = Binomial<Tp>::get(n);
std::vector<Tp> res(n);
res[0] = c;
for (int i = 1; i < n; ++i) res[i] = a[i - 1] * bin.inv(i);
return res;
}
template<typename Tp> inline std::vector<Tp> fps_log(const std::vector<Tp> &a, int n) {
return integr(fps_div(deriv(a), a, n - 1));
}
template<typename Tp> inline std::vector<Tp> fps_exp(const std::vector<Tp> &a, int n) {
if (n <= 0) return {};
assert(a.empty() || a[0] == 0);
return semi_relaxed_convolution(
deriv(a),
[bin = Binomial<Tp>::get(n)](int n, auto &&c) {
return n == 0 ? Tp(1) : c[n - 1] * bin.inv(n);
},
n);
}
template<typename Tp> inline std::vector<Tp> fps_pow(std::vector<Tp> a, long long e, int n) {
if (n <= 0) return {};
if (e == 0) {
std::vector<Tp> res(n);
res[0] = 1;
return res;
}
const int o = order(a);
if (o < 0 || o > n / e || (o == n / e && n % e == 0)) return std::vector<Tp>(n);
if (o != 0) a.erase(a.begin(), a.begin() + o);
const Tp ia0 = a[0].inv();
const Tp a0e = a[0].pow(e);
const Tp me = e;
for (int i = 0; i < (int)a.size(); ++i) a[i] *= ia0;
a = fps_log(a, n - o * e);
for (int i = 0; i < (int)a.size(); ++i) a[i] *= me;
a = fps_exp(a, n - o * e);
for (int i = 0; i < (int)a.size(); ++i) a[i] *= a0e;
a.insert(a.begin(), o * e, Tp(0));
return a;
}
#line 2 "fps_composition.hpp"
#line 10 "fps_composition.hpp"
// returns f(g) mod x^n
// see:
// [1]: Yasunori Kinoshita, Baitian Li. Power Series Composition in Near-Linear Time.
// https://arxiv.org/abs/2404.05177
template<typename Tp>
inline std::vector<Tp> composition(const std::vector<Tp> &f, const std::vector<Tp> &g, int n) {
if (n <= 0) return {};
if (g.empty()) {
std::vector<Tp> res(n);
if (!f.empty()) res[0] = f[0];
return res;
}
// [y^(-1)] (f(y) / (-g(x) + y)) mod x^n in R[x]((y^(-1)))
auto rec = [g0 = g[0]](auto &&rec, const std::vector<Tp> &P, const std::vector<Tp> &Q, int d,
int n) {
if (n == 1) {
std::vector<Tp> invQ(d + 1);
auto &&bin = Binomial<Tp>::get(d * 2);
Tp gg = 1;
for (int i = 0; i <= d; ++i) invQ[d - i] = bin.binom(d + i - 1, d - 1) * gg, gg *= g0;
// invQ[i] = [y^(-2d + i)]Q^(-1)
// P[0,d-1] * invQ[-2d,-d] => [0,d-1] * [0,d]
// take [-d,-1] => take [d,2d-1]
auto PinvQ = convolution(P, invQ);
PinvQ.erase(PinvQ.begin(), PinvQ.begin() + d);
PinvQ.resize(d);
return PinvQ;
}
std::vector<Tp> dftQ(d * n * 4);
for (int i = 0; i < d; ++i)
for (int j = 0; j < n; ++j) dftQ[i * (n * 2) + j] = Q[i * n + j];
dftQ[d * n * 2] = 1;
fft(dftQ);
std::vector<Tp> V(d * n * 2);
for (int i = 0; i < d * n * 4; i += 2) V[i / 2] = dftQ[i] * dftQ[i + 1];
inv_fft(V);
V[0] -= 1;
for (int i = 1; i < d * 2; ++i)
for (int j = 0; j < n / 2; ++j) V[i * (n / 2) + j] = V[i * n + j];
V.resize(d * n);
const auto T = rec(rec, P, std::move(V), d * 2, n / 2);
std::vector<Tp> dftT(d * n * 2);
for (int i = 0; i < d * 2; ++i)
for (int j = 0; j < n / 2; ++j) dftT[i * n + j] = T[i * (n / 2) + j];
fft(dftT);
std::vector<Tp> U(d * n * 4);
for (int i = 0; i < d * n * 4; i += 2) {
U[i] = dftT[i / 2] * dftQ[i + 1];
U[i + 1] = dftT[i / 2] * dftQ[i];
}
inv_fft(U);
// [-2d,d-1] => [0,3d-1]
// take [-d,-1] => take [d,2d-1]
for (int i = 0; i < d; ++i)
for (int j = 0; j < n; ++j) U[i * n + j] = U[(i + d) * (n * 2) + j];
U.resize(d * n);
return U;
};
const int k = fft_len(std::max<int>(n, f.size()));
std::vector<Tp> Q(k);
for (int i = 0; i < std::min<int>(k, g.size()); ++i) Q[i] = -g[i];
auto res = rec(rec, f, Q, 1, k);
res.resize(n);
return res;
}
// returns [x^k]gf^0, [x^k]gf, ..., [x^k]gf^(n-1)
// see:
// [1]: noshi91. FPS の合成と逆関数、冪乗の係数列挙 Θ(n (log(n))^2)
// https://noshi91.hatenablog.com/entry/2024/03/16/224034
template<typename Tp> inline std::vector<Tp>
enum_kth_term_of_power(const std::vector<Tp> &f, const std::vector<Tp> &g, int k, int n) {
if (k < 0 || n <= 0) return {};
if (f.empty()) {
std::vector<Tp> res(n);
if (k < (int)g.size()) res[0] = g[k];
return res;
}
// [x^k] (g(x) / (-f(x) + y)) in R[x]((y^(-1)))
std::vector<Tp> P(g), Q(k + 1);
P.resize(k + 1);
for (int i = 0; i < std::min<int>(k + 1, f.size()); ++i) Q[i] = -f[i];
int d = 1;
for (; k; d *= 2, k /= 2) {
const int len = fft_len((d * 2) * ((k + 1) * 2) - 1);
std::vector<Tp> dftP(len), dftQ(len);
for (int i = 0; i < d; ++i)
for (int j = 0; j <= k; ++j) {
dftP[i * ((k + 1) * 2) + j] = P[i * (k + 1) + j];
dftQ[i * ((k + 1) * 2) + j] = Q[i * (k + 1) + j];
}
dftQ[d * (k + 1) * 2] = 1;
fft(dftP);
fft(dftQ);
P.resize(len / 2);
Q.resize(len / 2);
if (k & 1) {
auto &&root = FftInfo<Tp>::get().inv_root(len / 2);
for (int i = 0; i < len; i += 2) {
P[i / 2] = (dftP[i] * dftQ[i + 1] - dftP[i + 1] * dftQ[i]).div_by_2() * root[i / 2];
Q[i / 2] = dftQ[i] * dftQ[i + 1];
}
} else {
for (int i = 0; i < len; i += 2) {
P[i / 2] = (dftP[i] * dftQ[i + 1] + dftP[i + 1] * dftQ[i]).div_by_2();
Q[i / 2] = dftQ[i] * dftQ[i + 1];
}
}
inv_fft(P);
inv_fft(Q);
if (d * (k + 1) * 4 >= len) Q[(d * (k + 1) * 4) % len] -= 1;
for (int i = 1; i < d * 2; ++i)
for (int j = 0; j <= k / 2; ++j) {
P[i * (k / 2 + 1) + j] = P[i * (k + 1) + j];
Q[i * (k / 2 + 1) + j] = Q[i * (k + 1) + j];
}
P.resize(d * 2 * (k / 2 + 1));
Q.resize(d * 2 * (k / 2 + 1));
}
std::vector<Tp> invQ(n + 1);
auto &&bin = Binomial<Tp>::get(d + n);
Tp ff = 1;
for (int i = 0; i <= n; ++i) invQ[n - i] = bin.binom(d + i - 1, d - 1) * ff, ff *= f[0];
// invQ[i] = [y^(-2d + i)]Q^(-1)
// P[0,d-1] * invQ[-(d+n),-d] => [0,d-1] * [0,n]
auto PinvQ = convolution(P, invQ);
// take [-n,-1] => take [d,d+n-1]
PinvQ.erase(PinvQ.begin(), PinvQ.begin() + d);
PinvQ.resize(n);
// output => [-1,-n] reverse
// before I just reverse it and mistaken something.
std::reverse(PinvQ.begin(), PinvQ.end());
return PinvQ;
}
// returns g s.t. f(g) = g(f) = x mod x^n
template<typename Tp> inline std::vector<Tp> reversion(std::vector<Tp> f, int n) {
if (n <= 0 || f.size() < 2) return {};
assert(order(f) == 1);
const auto if1 = f[1].inv();
if (n == 1) return {Tp(0)};
f.resize(n);
Tp ff = 1;
for (int i = 1; i < n; ++i) f[i] *= ff *= if1;
auto a = enum_kth_term_of_power(f, {Tp(1)}, n - 1, n);
auto &&bin = Binomial<Tp>::get(n);
for (int i = 1; i < n; ++i) a[i] *= (n - 1) * bin.inv(i);
auto b = fps_pow(std::vector(a.rbegin(), a.rend() - 1), Tp(1 - n).inv().val(), n - 1);
for (int i = 0; i < n - 1; ++i) b[i] *= if1;
b.insert(b.begin(), Tp(0));
return b;
}
#line 2 "fps_polya.hpp"
#line 7 "fps_polya.hpp"
// returns SEQ(A)=1/(1-a)
template<typename Tp> inline std::vector<Tp> polya_q(std::vector<Tp> a, int n) {
if (n <= 0) return {};
a.resize(n);
assert(a[0] == 0);
a[0] = 1;
for (int i = 1; i < n; ++i) a[i] = -a[i];
return inv(a, n);
}
// returns MSET(A)=exp(a(x)+a(x^2)/2+a(x^3)/3+...)
template<typename Tp> inline std::vector<Tp> polya_exp(std::vector<Tp> a, int n) {
if (n <= 0) return {};
a.resize(n);
assert(a[0] == 0);
auto &&bin = Binomial<Tp>::get(n);
for (int i = n - 1; i > 0; --i)
for (int j = 2; i * j < n; ++j) a[i * j] += a[i] * bin.inv(j);
return fps_exp(a, n);
}
// returns PSET(A)=exp(a(x)-a(x^2)/2+a(x^3)/3-...)
template<typename Tp> inline std::vector<Tp> polya_exp_m(std::vector<Tp> a, int n) {
if (n <= 0) return {};
a.resize(n);
assert(a[0] == 0);
auto &&bin = Binomial<Tp>::get(n);
for (int i = n - 1; i > 0; --i)
for (int j = 2; i * j < n; ++j)
if (j & 1) {
a[i * j] += a[i] * bin.inv(j);
} else {
a[i * j] -= a[i] * bin.inv(j);
}
return fps_exp(a, n);
}
#line 2 "poly_basic.hpp"
#line 7 "poly_basic.hpp"
#include <array>
#line 10 "poly_basic.hpp"
template<typename Tp> inline int degree(const std::vector<Tp> &a) {
int n = (int)a.size() - 1;
while (n >= 0 && a[n] == 0) --n;
return n;
}
template<typename Tp> inline void shrink(std::vector<Tp> &a) { a.resize(degree(a) + 1); }
template<typename Tp> inline std::vector<Tp> taylor_shift(std::vector<Tp> a, Tp c) {
const int n = a.size();
auto &&bin = Binomial<Tp>::get(n);
for (int i = 0; i < n; ++i) a[i] *= bin.factorial(i);
Tp cc = 1;
std::vector<Tp> b(n);
for (int i = 0; i < n; ++i) {
b[i] = cc * bin.inv_factorial(i);
cc *= c;
}
std::reverse(a.begin(), a.end());
auto ab = convolution(a, b);
ab.resize(n);
std::reverse(ab.begin(), ab.end());
for (int i = 0; i < n; ++i) ab[i] *= bin.inv_factorial(i);
return ab;
}
// returns (quotient, remainder)
// O(deg(Q)deg(B))
template<typename Tp> inline std::array<std::vector<Tp>, 2>
euclidean_div_naive(const std::vector<Tp> &A, const std::vector<Tp> &B) {
const int degA = degree(A);
const int degB = degree(B);
assert(degB >= 0);
const int degQ = degA - degB;
if (degQ < 0) return {std::vector<Tp>{Tp(0)}, A};
std::vector<Tp> Q(degQ + 1), R = A;
const auto inv = B[degB].inv();
for (int i = degQ, n = degA; i >= 0; --i)
if ((Q[i] = R[n--] * inv) != 0)
for (int j = 0; j <= degB; ++j) R[i + j] -= Q[i] * B[j];
R.resize(degB);
return {Q, R};
}
// O(min(deg(Q)^2,deg(Q)deg(B)))
template<typename Tp> inline std::vector<Tp>
euclidean_div_quotient_naive(const std::vector<Tp> &A, const std::vector<Tp> &B) {
const int degA = degree(A);
const int degB = degree(B);
assert(degB >= 0);
const int degQ = degA - degB;
if (degQ < 0) return {Tp(0)};
const auto inv = B[degB].inv();
std::vector<Tp> Q(degQ + 1);
for (int i = 0; i <= degQ; ++i) {
for (int j = 1; j <= std::min(i, degB); ++j) Q[degQ - i] += B[degB - j] * Q[degQ - i + j];
Q[degQ - i] = (A[degA - i] - Q[degQ - i]) * inv;
}
return Q;
}
// returns (quotient, remainder)
template<typename Tp> inline std::array<std::vector<Tp>, 2>
euclidean_div(const std::vector<Tp> &A, const std::vector<Tp> &B) {
const int degA = degree(A);
const int degB = degree(B);
assert(degB >= 0);
// A = Q*B + R => A/B = Q + R/B in R((x^(-1)))
const int degQ = degA - degB;
if (degQ < 0) return {std::vector<Tp>{Tp(0)}, A};
if (degQ < 60 || degB < 60) return euclidean_div_naive(A, B);
auto Q = fps_div(std::vector(A.rend() - (degA + 1), A.rend()),
std::vector(B.rend() - (degB + 1), B.rend()), degQ + 1);
std::reverse(Q.begin(), Q.end());
// returns a mod (x^n-1)
auto make_cyclic = [](const std::vector<Tp> &a, int n) {
assert((n & (n - 1)) == 0);
std::vector<Tp> b(n);
for (int i = 0; i < (int)a.size(); ++i) b[i & (n - 1)] += a[i];
return b;
};
const int len = fft_len(std::max(degB, 1));
const auto cyclicA = make_cyclic(A, len);
auto cyclicB = make_cyclic(B, len);
auto cyclicQ = make_cyclic(Q, len);
fft(cyclicQ);
fft(cyclicB);
for (int i = 0; i < len; ++i) cyclicQ[i] *= cyclicB[i];
inv_fft(cyclicQ);
// R = A - QB mod (x^n-1) (n >= degB)
std::vector<Tp> R(degB);
for (int i = 0; i < degB; ++i) R[i] = cyclicA[i] - cyclicQ[i];
return {Q, R};
}
template<typename Tp>
inline std::vector<Tp> euclidean_div_quotient(const std::vector<Tp> &A, const std::vector<Tp> &B) {
const int degA = degree(A);
const int degB = degree(B);
assert(degB >= 0);
// A = Q*B + R => A/B = Q + R/B in R((x^(-1)))
const int degQ = degA - degB;
if (degQ < 0) return {Tp(0)};
if (std::min(degQ, degB) < 60) return euclidean_div_quotient_naive(A, B);
auto Q = fps_div(std::vector(A.rend() - (degA + 1), A.rend()),
std::vector(B.rend() - (degB + 1), B.rend()), degQ + 1);
std::reverse(Q.begin(), Q.end());
return Q;
}
#line 2 "pow_table.hpp"
#line 4 "pow_table.hpp"
// returns 0^e, 1^e, ..., (n-1)^e
template<typename Tp> inline std::vector<Tp> pow_table(int e, int n) {
if (n <= 0) return {};
std::vector<bool> is_comp(n);
std::vector<int> p;
std::vector<Tp> res(n);
res[0] = (e == 0 ? Tp(1) : Tp(0)); // 0^0=1
if (n >= 2) res[1] = Tp(1);
for (int i = 2; i < n; ++i) {
if (!is_comp[i]) res[i] = Tp(p.emplace_back(i)).pow(e);
for (int j = 0; j < (int)p.size() && i * p[j] < n; ++j) {
is_comp[i * p[j]] = true;
res[i * p[j]] = res[i] * res[p[j]];
if (i % p[j] == 0) break;
}
}
return res;
}
#line 11 "famous_sequence.hpp"
// returns P([0..n)) s.t. P(n)=#(ways writing an integer n as sum of positive integers)
// see: https://mathworld.wolfram.com/PartitionFunctionP.html
// see also Pentagonal number
template<typename Tp> inline std::vector<Tp> partition_function(int n) {
assert(n >= 0);
std::vector<Tp> I(n);
for (int i = 1; i < n; ++i) I[i] = 1;
return polya_exp(I, n);
}
// returns |s(n,0)|, ..., |s(n,n)|
// unsigned Stirling numbers of the first kind
template<typename Tp> inline std::vector<Tp> unsigned_stirling_numbers_1st_row(int n) {
assert(n >= 0);
if (n == 0) return {Tp(1)};
int mask = 1 << 30;
while ((mask & n) == 0) mask >>= 1;
std::vector<Tp> res{Tp(0), Tp(1)};
for (int d = 1; d != n;) {
res = convolution(res, taylor_shift(res, Tp(d)));
d <<= 1;
if ((mask >>= 1) & n) {
res = convolution(res, std::vector<Tp>{Tp(d), Tp(1)});
d |= 1;
}
}
return res;
}
// returns |s(0,k)|, ..., |s(n-1,k)|
template<typename Tp> inline std::vector<Tp> unsigned_stirling_numbers_1st_column(int k, int n) {
assert(n >= 0);
auto &&bin = Binomial<Tp>::get(n);
std::vector<Tp> I(n);
for (int i = 1; i < n; ++i) I[i] = bin.inv(i);
auto res = fps_pow(I, k, n);
Tp v = 1;
for (int i = 1; i <= k; ++i) v *= i;
v = v.inv();
for (int i = 0; i < n; ++i) res[i] *= bin.factorial(i) * v;
return res;
}
// returns s(0,k), ..., s(n-1,k)
template<typename Tp> inline std::vector<Tp> signed_stirling_numbers_1st_column(int k, int n) {
auto S = unsigned_stirling_numbers_1st_column<Tp>(k, n);
for (int i = 0; i < n; ++i)
if ((k - i) & 1) S[i] = -S[i];
return S;
}
// returns s(n,0), ..., s(n,n)
template<typename Tp> inline std::vector<Tp> signed_stirling_numbers_1st_row(int n) {
auto S = unsigned_stirling_numbers_1st_row<Tp>(n);
for (int i = 0; i <= n; ++i)
if ((n - i) & 1) S[i] = -S[i];
return S;
}
// returns S(n,0), ..., S(n,n)
template<typename Tp> inline std::vector<Tp> stirling_numbers_2nd_row(int n) {
assert(n >= 0);
std::vector<Tp> res(n + 1), R(n + 1);
auto &&bin = Binomial<Tp>::get(n + 1);
const auto T = pow_table<Tp>(n, n + 1);
for (int i = 0; i <= n; ++i) {
R[i] = T[i] * (res[i] = bin.inv_factorial(i));
if (i & 1) res[i] = -res[i];
}
res = convolution(res, R);
res.resize(n + 1);
return res;
}
// returns S(0,k), ..., S(n-1,k)
template<typename Tp> inline std::vector<Tp> stirling_numbers_2nd_column(int k, int n) {
assert(n >= 0);
if (n == 0) return {};
if (n <= k) return std::vector<Tp>(n);
if (k == 0) {
std::vector<Tp> res(n);
res[0] = 1;
return res;
}
int mask = 1 << 30;
while ((mask & k) == 0) mask >>= 1;
std::vector<Tp> res{Tp(-1), Tp(1)};
for (int d = 1; d != k;) {
res = convolution(res, taylor_shift(res, -Tp(d)));
d <<= 1;
if ((mask >>= 1) & k) res = convolution(res, std::vector<Tp>{-Tp(d |= 1), Tp(1)});
}
res = fps_inv(std::vector(res.rbegin(), res.rend()), n - k);
res.insert(res.begin(), k, Tp(0));
return res;
}
// Eulerian numbers (OEIS) https://oeis.org/wiki/Eulerian_numbers,_triangle_of
// returns A(n,0), ..., A(n,n)
template<typename Tp> inline std::vector<Tp> eulerian_numbers_row(int n) {
std::vector<Tp> A(n + 1);
for (int i = 0; i <= n; ++i) A[i] = Tp(i + 1).pow(n);
auto AA = convolution(A, fps_pow(std::vector{Tp(1), Tp(-1)}, n + 1, n + 1));
AA.resize(n + 1);
return AA;
}
// returns A(0,k), ..., A(m-1,k)
// see:
// [1]: Entropy Increaser. 平移指数基变换.
// https://blog.csdn.net/EI_Captain/article/details/108586699
template<typename Tp> inline std::vector<Tp> eulerian_numbers_column(int k, int m) {
std::vector<Tp> A(k + 1), B(k + 1);
auto &&bin = Binomial<Tp>::get(std::max(k + 1, m));
for (int i = 0; i <= k; ++i) A[k - i] = Tp(-i - 1).pow(k - i) * bin.inv_factorial(k - i);
for (int i = 1; i <= k; ++i) B[k - i] = Tp(-i).pow(k - i) * bin.inv_factorial(k - i);
std::vector<Tp> xe_neg_x(m); // xe^(-x)
for (int i = 1; i < m; ++i) {
xe_neg_x[i] = bin.inv_factorial(i - 1);
if ((i - 1) & 1) xe_neg_x[i] = -xe_neg_x[i];
}
auto AA = convolution(composition(A, xe_neg_x, m), fps_exp(std::vector{Tp(0), Tp(k + 1)}, m));
auto BB = convolution(composition(B, xe_neg_x, m), fps_exp(std::vector{Tp(0), Tp(k)}, m));
for (int i = 0; i < m; ++i) AA[i] = (AA[i] - BB[i]) * bin.factorial(i);
AA.resize(m);
return AA;
}
template<typename Tp> inline std::vector<Tp> bell_numbers(int n) {
auto &&bin = Binomial<Tp>::get(n);
std::vector<Tp> ex(n);
for (int i = 1; i < n; ++i) ex[i] = bin.inv_factorial(i);
auto res = fps_exp(ex, n);
for (int i = 0; i < n; ++i) res[i] *= bin.factorial(i);
return res;
}
#line 2 "modint.hpp"
#include <iostream>
#line 5 "modint.hpp"
// clang-format off
template<unsigned Mod> class ModInt {
static_assert((Mod >> 31) == 0, "`Mod` must less than 2^(31)");
template<typename Int>
static std::enable_if_t<std::is_integral_v<Int>, unsigned> safe_mod(Int v) { using D = std::common_type_t<Int, unsigned>; return (v %= (int)Mod) < 0 ? (D)(v + (int)Mod) : (D)v; }
struct PrivateConstructor {} static inline private_constructor{};
ModInt(PrivateConstructor, unsigned v) : v_(v) {}
unsigned v_;
public:
static unsigned mod() { return Mod; }
static ModInt from_raw(unsigned v) { return ModInt(private_constructor, v); }
static ModInt zero() { return from_raw(0); }
static ModInt one() { return from_raw(1); }
bool is_zero() const { return v_ == 0; }
bool is_one() const { return v_ == 1; }
ModInt() : v_() {}
template<typename Int, typename std::enable_if_t<std::is_signed_v<Int>, int> = 0> ModInt(Int v) : v_(safe_mod(v)) {}
template<typename Int, typename std::enable_if_t<std::is_unsigned_v<Int>, int> = 0> ModInt(Int v) : v_(v % Mod) {}
unsigned val() const { return v_; }
ModInt operator-() const { return from_raw(v_ == 0 ? v_ : Mod - v_); }
ModInt pow(long long e) const { if (e < 0) return inv().pow(-e); for (ModInt x(*this), res(from_raw(1));; x *= x) { if (e & 1) res *= x; if ((e >>= 1) == 0) return res; }}
ModInt inv() const { int x1 = 1, x3 = 0, a = val(), b = Mod; while (b) { const int q = a / b, x1_old = x1, a_old = a; x1 = x3, x3 = x1_old - x3 * q, a = b, b = a_old - b * q; } return from_raw(x1 < 0 ? x1 + (int)Mod : x1); }
template<bool Odd = (Mod & 1)> std::enable_if_t<Odd, ModInt> div_by_2() const { if (v_ & 1) return from_raw((v_ + Mod) >> 1); return from_raw(v_ >> 1); }
ModInt &operator+=(const ModInt &a) { if ((v_ += a.v_) >= Mod) v_ -= Mod; return *this; }
ModInt &operator-=(const ModInt &a) { if ((v_ += Mod - a.v_) >= Mod) v_ -= Mod; return *this; }
ModInt &operator*=(const ModInt &a) { v_ = (unsigned long long)v_ * a.v_ % Mod; return *this; }
ModInt &operator/=(const ModInt &a) { return *this *= a.inv(); }
ModInt &operator++() { return *this += one(); }
ModInt operator++(int) { ModInt o(*this); *this += one(); return o; }
ModInt &operator--() { return *this -= one(); }
ModInt operator--(int) { ModInt o(*this); *this -= one(); return o; }
friend ModInt operator+(const ModInt &a, const ModInt &b) { return ModInt(a) += b; }
friend ModInt operator-(const ModInt &a, const ModInt &b) { return ModInt(a) -= b; }
friend ModInt operator*(const ModInt &a, const ModInt &b) { return ModInt(a) *= b; }
friend ModInt operator/(const ModInt &a, const ModInt &b) { return ModInt(a) /= b; }
friend bool operator==(const ModInt &a, const ModInt &b) { return a.v_ == b.v_; }
friend bool operator!=(const ModInt &a, const ModInt &b) { return a.v_ != b.v_; }
friend std::istream &operator>>(std::istream &a, ModInt &b) { int v; a >> v; b.v_ = safe_mod(v); return a; }
friend std::ostream &operator<<(std::ostream &a, const ModInt &b) { return a << b.val(); }
};
// clang-format on
#line 7 "test/enumerative_combinatorics/stirling_number_of_the_first_kind_fixed_k.0.test.cpp"
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
using mint = ModInt<998244353>;
int n, k;
std::cin >> n >> k;
const auto S = signed_stirling_numbers_1st_column<mint>(k, n + 1);
for (int i = k; i <= n; ++i) std::cout << S[i] << ' ';
return 0;
}