//#undef LOCAL #include using namespace std; using uint = unsigned int; using ll = long long; using ull = unsigned long long; template using V = vector; template using VV = V>; constexpr ll TEN(int n) { return (n == 0) ? 1 : 10 * TEN(n-1); } #define FOR(i, a, b) for(int i=(int)(a);i<(int)(b);i++) #define rep(i,N) for(int i=0;i<(int)(N);i++) #define rep1(i,N) for(int i=1;i<=(int)(N);i++) #define fs first #define sc second #define eb emplace_back #define pb eb #define all(x) x.begin(),x.end() template void chmin(T& t, const U& u) { if (t > u) t = u; } template void chmax(T& t, const U& u) { if (t < u) t = u; } #ifdef LOCAL #define show(x) cerr << __LINE__ << " : " << #x << " = " << (x) << endl #else #define show(x) true #endif template ostream& operator<<(ostream& os, const pair& p) { return os << "P(" << p.first << ", " << p.second << ")"; } template ostream& operator<<(ostream& os, const V& v) { os << "["; for (auto d : v) os << d << ", "; return os << "]"; } // cin.tie(nullptr); // ios::sync_with_stdio(false); // cout << fixed << setprecision(20); template struct ModInt { using M = ModInt; const static M G; uint v; ModInt(ll _v = 0) { set_v(_v % MD + MD); } M& set_v(uint _v) { v = (_v < MD) ? _v : _v - MD; return *this; } explicit operator bool() const { return v != 0; } M operator-() const { return M() - *this; } M operator+(const M& r) const { return M().set_v(v + r.v); } M operator-(const M& r) const { return M().set_v(v + MD - r.v); } M operator*(const M& r) const { return M().set_v(ull(v) * r.v % MD); } M operator/(const M& r) const { return *this * r.inv(); } M& operator+=(const M& r) { return *this = *this + r; } M& operator-=(const M& r) { return *this = *this - r; } M& operator*=(const M& r) { return *this = *this * r; } M& operator/=(const M& r) { return *this = *this / r; } bool operator==(const M& r) const { return v == r.v; } M pow(ll n) const { M x = *this, r = 1; while (n) { if (n & 1) r *= x; x *= x; n >>= 1; } return r; } M inv() const { return pow(MD - 2); } friend ostream& operator<<(ostream& os, const M& r) { return os << r.v;} }; template<> const ModInt<998244353> ModInt<998244353>::G = 3; // 適切な原始根を指定します using Mint = ModInt<998244353>; ll modpow(ll base, ll exp, ll mod) { ll res = 1; while (exp > 0) { if (exp % 2 == 1) res = res * base % mod; base = base * base % mod; exp /= 2; } return res; } //x^2=N(mod p)となるxを返す(存在しないなら-1) //modpow(a,b,p)はa^b(mod p) ll Tonelli_Shanks(ll N, ll p) { N%=p; if(p==2){ return N; } if (modpow(N, p >> 1, p) == p - 1) { return -1; } else if (p % 4 == 3) { return modpow(N, (p + 1) / 4, p); } else { ll n = 1; for (; n < p; n++) { if (modpow(n, p >> 1, p) == p - 1) { break; } } ll pp = p - 1; int c = 0; while (pp % 2 == 0) { pp /= 2; c++; } ll s = modpow(N, pp, p); ll r = modpow(N, (pp + 1) / 2, p); for (int i = c - 2; i >= 0; --i) { if (modpow(s, 1LL << i, p) == p - 1) { s = s * modpow(n, p >> (1 + i), p) % p; r = r * modpow(n, p >> (2 + i), p) % p; } } return r; } } void nft(bool type, V& a) { int n = int(a.size()), s = 0; while ((1 << s) < n) s++; assert(1 << s == n); static V ep, iep; while (int(ep.size()) <= s) { ep.push_back(Mint::G.pow(Mint(-1).v / (1 << ep.size()))); iep.push_back(ep.back().inv()); } V b(n); for (int i = 1; i <= s; i++) { int w = 1 << (s - i); Mint base = type ? iep[i] : ep[i], now = 1; for (int y = 0; y < n / 2; y += w) { for (int x = 0; x < w; x++) { auto l = a[y << 1 | x]; auto r = now * a[y << 1 | x | w]; b[y | x] = l + r; b[y | x | n >> 1] = l - r; } now *= base; } swap(a, b); } } template V multiply(const V& a, const V& b) { int n = int(a.size()), m = int(b.size()); if (!n || !m) return {}; if (min(n, m) <= 8) { V ans(n + m - 1); for (int i = 0; i < n; i++) for (int j = 0; j < m; j++) ans[i + j] += a[i] * b[j]; return ans; } int lg = 0; while ((1 << lg) < n + m - 1) lg++; int z = 1 << lg; auto a2 = a, b2 = b; a2.resize(z); b2.resize(z); nft(false, a2); nft(false, b2); for (int i = 0; i < z; i++) a2[i] *= b2[i]; nft(true, a2); a2.resize(n + m - 1); Mint iz = Mint(z).inv(); for (int i = 0; i < n + m - 1; i++) a2[i] *= iz; return a2; } template struct Poly { vector v; Poly(const vector& _v = {}) : v(_v) { shrink(); } void shrink() { while (v.size() && !v.back()) v.pop_back(); } int size() const { return int(v.size()); } D freq(int p) const { return (p < size()) ? v[p] : D(0); } Poly operator+(const Poly& r) const { auto n = max(size(), r.size()); vector res(n); for (int i = 0; i < n; i++) res[i] = freq(i) + r.freq(i); return res; } Poly operator-(const Poly& r) const { int n = max(size(), r.size()); vector res(n); for (int i = 0; i < n; i++) res[i] = freq(i) - r.freq(i); return res; } Poly operator*(const Poly& r) const { return {multiply(v, r.v)}; } Poly operator*(const D& r) const { int n = size(); vector res(n); for (int i = 0; i < n; i++) res[i] = v[i] * r; return res; } Poly operator/(const D &r) const{ return *this * r.inv(); } Poly operator/(const Poly& r) const { if (size() < r.size()) return {{}}; int n = size() - r.size() + 1; return (rev().pre(n) * r.rev().inv(n)).rev(n); //変更 } Poly operator%(const Poly& r) const { return *this - *this / r * r; } Poly operator<<(int s) const { vector res(size() + s); for (int i = 0; i < size(); i++) res[i + s] = v[i]; return res; } Poly operator>>(int s) const { if (size() <= s) return Poly(); vector res(size() - s); for (int i = 0; i < size() - s; i++) res[i] = v[i + s]; return res; } Poly& operator+=(const Poly& r) { return *this = *this + r; } Poly& operator-=(const Poly& r) { return *this = *this - r; } Poly& operator*=(const Poly& r) { return *this = *this * r; } Poly& operator*=(const D& r) { return *this = *this * r; } Poly& operator/=(const Poly& r) { return *this = *this / r; } Poly& operator/=(const D &r) {return *this = *this/r;} Poly& operator%=(const Poly& r) { return *this = *this % r; } Poly& operator<<=(const size_t& n) { return *this = *this << n; } Poly& operator>>=(const size_t& n) { return *this = *this >> n; } Poly pre(int le) const { return {{v.begin(), v.begin() + min(size(), le)}}; } Poly rev(int n = -1) const { vector res = v; if (n != -1) res.resize(n); reverse(res.begin(), res.end()); return res; } Poly diff() const { vector res(max(0, size() - 1)); for (int i = 1; i < size(); i++) res[i - 1] = freq(i) * i; return res; } Poly inte() const { vector res(size() + 1); for (int i = 0; i < size(); i++) res[i + 1] = freq(i) / (i + 1); return res; } // f * f.inv() = 1 + g(x)x^m Poly inv(int m) const { Poly res = Poly({D(1) / freq(0)}); for (int i = 1; i < m; i *= 2) { res = (res * D(2) - res * res * pre(2 * i)).pre(2 * i); } return res.pre(m); } Poly exp(int n) const { assert(freq(0) == 0); Poly f({1}), g({1}); for (int i = 1; i < n; i *= 2) { g = (g * 2 - f * g * g).pre(i); Poly q = diff().pre(i - 1); Poly w = (q + g * (f.diff() - f * q)).pre(2 * i - 1); f = (f + f * (*this - w.inte()).pre(2 * i)).pre(2 * i); } return f.pre(n); } Poly log(int n) const { assert(freq(0) == 1); auto f = pre(n); return (f.diff() * f.inv(n - 1)).pre(n - 1).inte(); } Poly sqrt(int n) const { assert(freq(0) == 1); Poly f = pre(n + 1); Poly g({1}); for (int i = 1; i < n; i *= 2) { g = (g + f.pre(2 * i) * g.inv(2 * i)) / 2; } return g.pre(n + 1); } //定数項が1である必要はない pair sqrt_arb(int n) const{ if(size()==0){ return {true,Poly()}; } int c=0; while(c*2>c*2; P/=x*x; P=P.sqrt(n-c); P<<=c; P*=x; return {true,P}; } // Poly power(ll k,int n){ if(!k){ return Poly({D(1)}); } if(!size()){ return Poly(); } int c=0; while(c(n-1)/k){ return Poly(); } Mint ic=freq(c),pc=freq(c); ic=ic.inv(); pc=pc.pow(k); int l=n-c*k; return (((((*this).pre(l+c)*ic>>c).log(l)*k).exp(l)*pc)<>= 1; } return r; } friend ostream& operator<<(ostream& os, const Poly& p) { if (p.size() == 0) return os << "0"; for (auto i = 0; i < p.size(); i++) { if (p.v[i]) { os << p.v[i] << "x^" << i; if (i != p.size() - 1) os << "+"; } } return os; } }; template struct MultiEval { using NP = MultiEval*; NP l, r; vector que; int sz; Poly mul; MultiEval(const vector& _que, int off, int _sz) : sz(_sz) { if (sz <= 100) { que = {_que.begin() + off, _que.begin() + off + sz}; mul = {{1}}; for (auto x : que) mul *= {{-x, 1}}; return; } l = new MultiEval(_que, off, sz / 2); r = new MultiEval(_que, off + sz / 2, sz - sz / 2); mul = l->mul * r->mul; } MultiEval(const vector& _que) : MultiEval(_que, 0, int(_que.size())) {} void query(const Poly& _pol, vector& res) const { if (sz <= 100) { for (auto x : que) { Mint sm = 0, base = 1; for (int i = 0; i < _pol.size(); i++) { sm += base * _pol.freq(i); base *= x; } res.push_back(sm); } return; } auto pol = _pol % mul; l->query(pol, res); r->query(pol, res); } vector query(const Poly& pol) const { vector res; query(pol, res); return res; } }; //rev()を取って-1倍すると、1+...の形で線形漸化式の分母になる template Poly berlekamp_massey(const vector& s) { int n = int(s.size()); vector b = {Mint(-1)}, c = {Mint(-1)}; Mint y = Mint(1); for (int ed = 1; ed <= n; ed++) { int l = int(c.size()), m = int(b.size()); Mint x = 0; for (int i = 0; i < l; i++) { x += c[i] * s[ed - l + i]; } b.push_back(0); m++; if (!x) continue; Mint freq = x / y; if (l < m) { // use b auto tmp = c; c.insert(begin(c), m - l, Mint(0)); for (int i = 0; i < m; i++) { c[m - 1 - i] -= freq * b[m - 1 - i]; } b = tmp; y = x; } else { // use c for (int i = 0; i < m; i++) { c[l - 1 - i] -= freq * b[m - 1 - i]; } } } return c; } // n/dのx^Nの係数 template D Bostan_Mori(Poly n, Poly d, ll N) { while (N) { Poly dd=Poly(d.v); for(int i=1;i>=1; for(int i=0;i>=1; } return n.size() ? n.freq(0) : D(0); } template D BMBM(vector A,ll N){ Poly d=berlekamp_massey(A).rev()*(-1); Poly n=d*Poly(A).pre(d.size()); return Bostan_Mori(n,d,N); } int main() { cin.tie(nullptr); ios::sync_with_stdio(false); int N; cin>>N; vector P={0}; vector dp={Mint(5).inv()*2,Mint(5).inv()*2,Mint(5).inv()}; for(int d=0;d<20;d++){ P.push_back(P.back()+dp[1]+dp[2]); dp={dp[0]/2+dp[1]/3,dp[0]/2+dp[1]/3+dp[2]*2/3,dp[1]/3+dp[2]/3}; } cout<