NTT

在模 \(p\) 意义下,若 \(p\) 的原根为 \(g\),可以用 \(g^{(p-1)/n}\) 代替 \(\omega_n^1\),它满足 FFT 中单位根的所有性质。

但必须 \(n\mid p-1\) 才行。我们通常把 \(n\) 补为 \(2\) 的幂。所以如果 \(p=r\times 2^m+1\),可以处理项数为 \(2^m\) 的多项式。

const int N = (1 << 20) + 1, P = 998244353, G = 3;
ll invG;
int tr[N], n;

inline ll poww(ll x, ll y = P - 2) {
	ll res = 1;
	for(; y; y >>= 1, x = x * x % P)
		if(y & 1) res = res * x % P;
	return res;
}

inline void madd(ll &x, const ll &y) {
	x += y;
	if(x >= P) x -= P;
}

inline ll mod(const ll x) {
	return x >= P ? x - P : x;
}

inline void cpy(ll *f, ll *g, int n) {
	rep(i, 0, n - 1) *(f + i) = *(g + i);
}

inline void clear(ll *f, int n) {
	rep(i, 0, n - 1) *(f + i) = 0;
}

inline void mul(ll *f, ll *g, int n) {
	rep(i, 0, n - 1) (f[i] *= g[i]) %= P;
}

void NTT(ll *f, int n, int fl) { // fl=1表示DFT,fl=0表示IDFT
	ll tmp, w1, buf;
	rep(i, 0, n - 1) tr[i] = (tr[i >> 1] >> 1) | (i & 1 ? (n >> 1) : 0);
	rep(i, 0, n - 1) if(i < tr[i]) swap(f[i], f[tr[i]]);
	for(rei lim = 2, lim2; lim <= n; lim <<= 1) {
		w1 = poww(fl ? G : invG, (P - 1) / lim);
		lim2 = lim >> 1;
		for(rei i = 0; i < n; i += lim) {
			buf = 1;
			rep(k, i, i + lim2 - 1) {
				tmp = buf * f[k + lim2] % P;
				f[k + lim2] = mod(f[k] - tmp + P);
				madd(f[k], tmp);
				(buf *= w1) %= P;
			}
		}
	}
	if(!fl) {
		tmp = poww(n);
		rep(i, 0, n) (f[i] *= tmp) %= P;
	}
}

1. 乘法逆元

题目链接

若 \(F(x)\) 为 \(n-1\) 次多项式,\(F(x)G(x)\equiv 1~(\bmod~x^n)\),则称 \(G(x)\) 是 \(F(x)\) 的乘法逆元。

如果 \(F(x)\) 有乘法逆元,则 \(F[0]\neq 0\) (\(F[i]\) 表示 \(F(x)\) 的 \(i\) 次项系数)。因为如果 \(F[0]=0\),无论 \(G[0]\) 是否为 \(0\),\(F(x)G(x)\) 的常数项都为 \(0\)。

在模 \(x\) 意义下,\(F(x)\) 的乘法逆元为 \(F[0]^{-1}\)。

考虑我们已经知道 \(F(x)\) 在 \(\bmod~x^{n/2}\) 意义下,逆元为 \(G^\prime(x)\) 即 \(F(x)G^\prime(x)\equiv 1~(\bmod~x^{n/2})\)。

现求满足 \(F(x)G(x)\equiv 1~(\bmod~x^n)\) 的 \(G(x)\)。两式相减可得 \(F(x)(G(x)-G^\prime(x))\equiv 1~(\bmod~x^{n/2})\)

显然 \(F(x)\not\equiv 0~(\bmod~x^{n/2})\),那么 \((G(x)-G^\prime(x))\equiv 0~(\bmod~x^{n/2})\)。平方可得 \(G(x)^2-2G(x)G^\prime(x)+G^\prime(x)^2\equiv 0~(\bmod~x^n)\)。

两边同乘 \(F(x)\) 得 \(G(x)^2F(x)-2F(x)G(x)G^\prime(x)+F(x)G^\prime(x)^2\equiv 0~(\bmod~x^n)\) 即 \(G(x)-2G^\prime(x)+F(x)G^\prime(x)^2\equiv 0~(\bmod~x^n)\)。

所以 \(G(x)\equiv 2G^\prime(x)-F(x)G^\prime(x)^2~(\bmod~x^n)\)。

倍增求解即可。

复杂度:\(T(n)=T(\dfrac{n}{2})+\operatorname{O}(n\log n)=\operatorname{O}(n\log n)\)。

void poly_inv(ll *f, int m) {
	static int n;
	static ll s[N], h[N], g[N];
	for(n = 1; n < m; n <<= 1) ;
	clear(s, n * 2);
	clear(g, n * 2);
	clear(h, n * 2);
	g[0] = poww(f[0]);
	for(rei lim = 2; lim <= n; lim <<= 1) {
		rep(i, 0, lim / 2 - 1) h[i] = mod(g[i] * 2);
		cpy(s, f, lim);
		NTT(g, lim * 2, 1);
		mul(g, g, lim * 2);
		NTT(s, lim * 2, 1);
		mul(g, s, lim * 2);
		NTT(g, lim * 2, 0);
		rep(i, 0, lim - 1) g[i] = mod(h[i] - g[i] + P);
		clear(g + lim, lim); // 一定要清空!!!!!!
	}
	cpy(f, g, m);
}

2. 开根

题目链接

在模 \(x\) 意义下,直接求二次剩余。本题保证 \(a_0=1\),就不需要求了,但加强版需要。

考虑我们已经知道在 \(\bmod~x^{n/2}\) 意义下,有 \(G^\prime(x)^2\equiv F(x)~(\bmod~x^{n/2})\)。

现求满足 \(G(x)^2\equiv F(x)~(\bmod~x^n)\) 的 \(G(x)\)。

显然有 \(G(x)\equiv G^\prime(x)~(\bmod~x^{n/2})\)。

移项,平方,得 \(G(x)^2-2G(x)G^\prime(x)+G^\prime(x)^2\equiv 0~(\bmod~x^n)\) 即 \(F(x)-2G(x)G^\prime(x)+G^\prime(x)^2\equiv 0~(\bmod~x^n)\)

整理得 \(G(x)\equiv \dfrac{1}{2}(G^\prime(x)+\dfrac{F(x)}{G^\prime(x)})\)。

倍增求解+求逆即可。

复杂度:\(T(n)=T(\dfrac{n}{2})+\operatorname{O}(n\log n)=\operatorname{O}(n\log n)\)。

void poly_sqrt(ll *f, int m) {
	static ll g[N], h[N], s[N];
	static int n;
	for(n = 1; n < m; n <<= 1) ;
	clear(s, n * 2);
	clear(g, n * 2);
	clear(h, n * 2);
	g[0] = 1; //由于保证a[0]=1,1^2=1。如果没有保证需求二次剩余
	for(rei lim = 2; lim <= n; lim <<= 1) {
		rep(i, 0, lim / 2 - 1) h[i] = g[i];
		clear(h + lim / 2, lim / 2);
		poly_inv(h, lim);
		NTT(h, lim * 2, 1);
		cpy(s, f, lim);
		NTT(s, lim * 2, 1);
		mul(h, s, lim * 2);
		NTT(h, lim * 2, 0);
		rep(i, 0, lim - 1) g[i] = (g[i] + h[i]) * inv2 % P;
		clear(g + lim, lim);
	}
	cpy(f, g, m);
}