https://beta.atcoder.jp/contests/agc005/tasks/agc005_f
FMT初実装です。
まず求めるものを式にすると
ans[K]=1/(K!)Σ(i=K...N)cnt[i]*(i!)/(i-K)!
となります。ただしcnt[i]はサイズがiである部分木の数です。
これを式変形すると、
ans[N-u]=1/((N-u)!)Σ(i=0...u)d[u-i]*e[i]となります。
ただしd[i]=cnt[N-i]*(N-i)!、e[i]=1/(i!)です。このようにdi,eiがK(またはu)に依存しない式が立てれれば、FMTが使えます。iとK-iの式で表わせる形があったらFFTを疑ったほうが良さそうですね。
FMTですが、FFTとほとんど同じです。代わりにω^n=1(mod p)を満たすようなωを探せばいいだけです。その際にpの原始根が必要になるんですが、与えられるpは因数分解すると綺麗な形になっているので、見つけるのもそんなに苦労はしないです。
今回はp=924844033ですが原子根の1つに5があります。
struct FMT { const int num_factor = 3; ll factor[3] = {2, 3, 7}; ll gen = 5; ll mod_pow(ll a, ll n) { if(n == 0) return 1; ll res = mod_pow(a, n / 2); if(n % 2 == 0) return res * res % mod; else return a * res % mod * res % mod; } bool check() { rep(i, 0, num_factor) { if(mod_pow(gen, (mod - 1) / factor[i]) == 1) { return false; } } return true; } void fmt(vector<ll>& v, bool rev = false) { //v.size() == (1 << x) int n = sz(v), i, j; for(i = 0, j = 1; j < n - 1; j++) { for(int k = n >> 1; k > (i ^= k); k /= 2); if(i > j) swap(v[i], v[j]); } for(int m = 2; m <= n; m *= 2) { ll root = mod_pow(gen, (mod - 1) / m); for(int i = 0; i < n; i += m) { ll w = 1; rep(j, 0, m / 2) { ll f0 = v[i + j], f1 = w * v[i + j + m / 2] % mod; v[i + j] = (f0 + f1) % mod; v[i + j + m / 2] = (mod + f0 - f1) % mod; MUL(w, root); } } } if(rev) rep(i, 0, n) MUL(v[i], mod_pow(n, mod - 2)); } vector<ll> get(const vector<ll>& A, const vector<ll>& B) { int n = 1; int len = sz(A) + sz(B) - 1; while(n < len) n *= 2; vector<ll> a(n, 0), b(n, 0); rep(i, 0, sz(A)) a[i] = A[i]; rep(i, 0, sz(B)) b[i] = B[i]; fmt(a); fmt(b); rep(i, 0, n) MUL(a[i], b[i]); reverse(++a.begin(), a.end()); fmt(a, true); vector<ll> res(len); rep(i, 0, len) res[i] = a[i]; return res; } }; int N; vi G[MAX_N]; int cnt[MAX_N]; int dfs(int v, int p = -1) { int res = 1; rep(i, 0, sz(G[v])) { int n = G[v][i]; if(n == p) continue; res += dfs(n, v); } if(p != -1) { cnt[res]++; cnt[N - res]++; } return res; } void solve() { cin >> N; C_init(N + 10); rep(i, 0, N - 1) { int a, b; cin >> a >> b; a--; b--; G[a].pb(b); G[b].pb(a); } dfs(0); vector<ll> D(N + 1), E(N + 1); rep(i, 0, N + 1) { D[i] = cnt[N - i] * fac[N - i] % mod; E[i] = fiv[i]; } FMT f; vector<ll> F = f.get(D, E); vector<ll> ans(N + 1); rep(i, 0, N) { ans[N - i] = (mod + N * C(N, i) % mod - fiv[N - i] * F[i] % mod) % mod; } rep(i, 1, N + 1) cout << ans[i] << "\n"; }