多项式牛顿迭代

Newton's Method 是牛顿提出的一种将非线性方程线性化的近似方法。它也可以运用在多项式中,求关于多项式的非线性方程在模意义下的解。

学习多项式牛顿迭代,要先了解泰勒级数的相关知识。

泰勒级数和麦克劳林级数

泰勒级数用无限项连加式来表示函数。一般地,对于一个光滑函数 f(x),有

f(x)=n=0+f(n)(a)i!(xa)n

这个等式被称为 f(x)a 处的泰勒展开式;等号右边的式子被称为 f(x)a 处的泰勒级数。在 0 处的泰勒展开式和泰勒级数也被称为麦克劳林展开式和麦克劳林级数。下面给出了一些常用的麦克劳林级数,有的在生成函数中常用:

(1x)1=n=0+xn

(1x)m=n=0+(n+m1n)xn

(1+x)m=n=0+(mn)xn

ln(1x)=n=1+xnn

ln(1+x)=n=1+(1)nxnn

expx=n=0+xnn!

泰勒展开式并非对于 f(x) 定义域内所有 x 都成立,但在 OI 中我们只关心展开式的系数,不关心 x 的值。了解了泰勒级数,我们可以开始学习 Newton's Method 了:

一般的 Newton's Method

Newton's Method 一般被用于求解非线性方程。它是这样求 f(x)=0 的根的:

  • 选取合适一个数作为 x0
  • f(x)x0 处展开,即 f(x)=n=0+f(n)(x0)i!(xx0)n
  • 取其常数项和线性项的系数,令其值为 0,即 f(x0)+f(x0)(xx0)=0
  • 解得这个近似方程的根 x1,并在 x1 处将 f(x) 泰勒展开,重复上述过程得到 x2,x3,x4,... 可以证明,每一个新解都更加接近 f(x)=0 的根
double f(double x);
double fd(double x);//fd 是 f 的导数,即 f'

double newtonMethod(double x0, int d){ //d 代表迭代次数
	//f(x_0)+fd(x0)*(x-x0)=0 -> x=-f(x0)/fd(x0)+x0
    while(d--) x0 = -f(x0) / fd(x0) + x0;
    return x0;
}

这样看来,我们只能无限逼近 f(x)=0 的根,而无法绝对准确地取到根。我们求多项式,是否也无法得到精确的解呢?担心是多余的,我们本来就不需要求“绝对精确”的解,只需要求模意义下的“精确”的解。

Newton's Method 在多项式中的 Methodology

求一个关于 x 的多项式 f(x),使得对于一给定的关于 f(x) 的函数 g(f(x))g(f(x))0(modxn) 成立。

这是 Newton's Method 求解的问题的通式。对于多项式 h(x),求逆时 g(f(x))=h(x)f1(x),开根时 g(f(x))=h(x)f2(x),求指数时 g(f(x))=h(x)lnf(x)。下面就讲述 Newton's Method 是如何逼近根的:

选择一个满足 g(f(x))0(modx1) 的多项式作为初始根。

设我们已经求出了模 xn2 下的根 f0(x),将 g(f(x))f0(x) 处泰勒展开:

g(f(x))i=0+g(i)(f0(x))i!(f(x)f0(x))i0(modxn)

f(x)=f0(x)+xn2h(x) 代入

i=0+g(i)(f0(x))i!(f0(x)+xn2h(x)f0(x))i0(modxn)

i=0+g(i)(f0(x))i!xin2hi(x)0(modxn)

其中,对于 xin2hi(x) 这一部分,显然当 i2 时有 xin2hi(x)0(modxn)。则在此处模 xn 意义下的泰勒级数,从第三项开始全部为 0,于是只需保留前两项:

g(f0(x))+g(f0(x))(xn2h(x))0(modxn)

解得

h(x)xn2g(f0(x))g(f0(x))(modxn)

h(x) 存在的情况下回代得 f(x)f0(x)g(f0(x))g(f0(x))(modxn),完成了一次迭代。如果迭代 a 次,我们就可以得到在模 x2a 意义下的解了。

有一个问题尚未解决:h(x) 是否总是存在?我们发现,xn2g(f0(x))g(f0(x)) 在模 xn 意义下存在,当且仅当它只包含 x 的自然数幂次项,而 g(f0(x))0(modxn2),于是在模 xn 意义下的 g(f0(x))g(f0(x)) 总是存在,且其前 n2 总是为零。于是 xn2g(f0(x))g(f0(x)) 总是一个只包含 x 的自然数幂的式子,h(x) 总是存在。

由上述过程可知,Newton's Method 可应用于解 g(f(x))=0xn 意义下的根,当且仅当 g(f(x)) 无限可微。g(f(x))0(modxn) 的根存在,当且仅当这个同余方程在模 x1 意义下的根 f0 存在,并且每个 f0 都对应着唯一的一个模 xn 的根。除此之外,对于第 a 次和第 a+1 次迭代的根 fa(x),fa+1(x),有 fa(x)fa+1(x)(modx2a)

下面将就几个具体的 g(f(x)) 讲解 Newton's Method 的应用。

Newton's Method 求多项式的逆

对于原多项式 h(x),它在模 xn 意义下的逆 f(x) 满足 h(x)g(x)0(modxn)。则 g(f(x)) 可以写作 g(f(x))=h(x)1f(x)。代入 Newton's Method 得

f(x)f0(x)h(x)1f0(x)f02(x)(modxn)

注意这里 g(f(x)) 不是关于 x 的复合函数,而是一个关于 f(x) 的函数。它的导数要视作 dg(f(x))df(x)

f(x)f0(x)(2f0(x)h(x))(modxn)

运用 FNTT,单次迭代复杂度为 O(nlogn)。运用主定理可得总时间复杂度为 T(n)=T(n2)+O(nlogn)=O(nlogn)。我们一般用倍增法实现代码。

void polyinv(int *f, const int *h, int n){
	int N = 1; while(N < n + n - 1) N <<= 1;
	static int d[maxn], g[maxn];
	memcpy(d, h, n * sizeof(int)), memset(d + n, 0, (N - n) * sizeof(int));
	memset(f, 0, N * sizeof(int)), memset(g, 0, N * sizeof(int)), f[0] = qpow(h[0], mod - 2);
	for(int w = 2; w / 2 < n; w <<= 1){
		memcpy(g, d, w * sizeof(int));
		for(int i = 0; i < w * 2; ++i) rev[i] = (rev[i >> 1] >> 1) | (i & 1 ? w : 0);
		dft(f, w << 1, 1), dft(g, w << 1, 1);
		for(int i = 0; i < w * 2; ++i) f[i] = (ll)f[i] * (2 - (ll)f[i] * g[i] % mod) % mod;
		dft(f, w << 1, -1);
		for(int i = 0, inv = qpow(w << 1, mod - 2); i < w; ++i) f[i] = (ll)f[i] * inv % mod;
		memset(f + w, 0, w * sizeof(int));
	}
	memset(f + n, 0, (N - n) * sizeof(int));
}

Newton's Method 求多项式平方根

对于多项式 h(x),它的平方根 f(x) 满足 f2(x)h(x)(modxn),我们可以写出 g(f(x))h(x)f2(x)(modxn)。代入 Newton's Method 得

f(x)f0(x)h(x)f02(x)2f0(x)f0(x)+12(h(x)f01(x)f0(x))12(f0(x)+f01(x)h(x))(modxn)

单次迭代仍为 O(nlogn),故总复杂度也为 O(nlogn)

void polysqrt(int *f, const int *h, int n){
	int N = 1; while(N < n + n - 1) N <<= 1;
	static int d[maxn], g[maxn], f_inv[maxn];
	memcpy(d, h, n * sizeof(int)), memset(d + n, 0, (N - n) * sizeof(int));
	memset(f, 0, N * sizeof(int)), memset(g, 0, N * sizeof(int)), f[0] = 1;
	for(int w = 2; w / 2 < n; w <<= 1){
		memcpy(g, d, w * sizeof(int)), polyinv(f_inv, f, w);
		for(int i = 0; i < w * 2; ++i) rev[i] = (rev[i >> 1] >> 1) | (i & 1 ? w : 0);
		dft(g, w << 1, 1), dft(f, w << 1, 1), dft(f_inv, w << 1, 1);
		for(int i = 0; i < w * 2; ++i) f[i] = (f[i] + (ll)f_inv[i] * g[i] % mod) % mod;
		dft(f, w << 1, -1);
		for(int i = 0, inv = qpow(w << 2, mod - 2); i < w; ++i) f[i] = (ll)f[i] * inv % mod;
		memset(f + w, 0, w * sizeof(int));
	}
	memset(f + n, 0, (N - n) * sizeof(int));
}

Newton's Method 求多项式指数函数

多项式的对数函数和指数函数都是用麦克劳林级数定义的。要用 Newton's Method 求多项式指数函数,得先会求多项式对数函数

对于多项式 h(x),它的指数函数 f(x)exph(x)(modxn) 满足 h(x)lnf(x)(modxn),我们可以写出 g(f(x))=h(x)lnf(x)。代入 Newton's Method 得

f(x)f0(x)h(x)lnf0(x)1f0(x)f0(x)+f0(x)(h(x)lnf0(x))f0(x)(1+h(x)lnf0(x))(modxn)

单次迭代仍为 O(nlogn),故总复杂度也为 O(nlogn)

void polyexp(int *f, const int *h, int n){
	int N = 1; while(N < n + n - 1) N <<= 1;
	static int d[maxn], g[maxn], lg[maxn]; memset(lg, 0, N * sizeof(int));
	memcpy(d, h, n * sizeof(int)), memset(d + n, 0, (N - n) * sizeof(int));
	memset(g, 0, N * sizeof(int)), memset(f, 0, N * sizeof(int)), f[0] = 1;
	for(int w = 2; w / 2 < n; w <<= 1){
		memcpy(g, d, w * sizeof(int)), polylog(lg, f, w);
		for(int i = 0; i < w * 2; ++i) rev[i] = (rev[i >> 1] >> 1) | (i & 1 ? w : 0);
		dft(g, w << 1, 1), dft(f, w << 1, 1), dft(lg, w << 1, 1);
		for(int i = 0; i < w * 2; ++i) f[i] = (ll)f[i] * (1 + g[i] - lg[i]) % mod;
		dft(f, w << 1, -1);
		for(int i = 0, inv = qpow(w << 1, mod - 2); i < w; ++i) f[i] = (ll)f[i] * inv % mod;
		memset(f + w, 0, w * sizeof(int));
	}
	memset(f + n, 0, (N - n) * sizeof(int));
}

一个有趣的事实是,要写多项式指数函数就必须先写一个多项式对数函数;要写多项式对数函数就必须先写一个多项式求逆......这些东西加起来有七八十行,说实话还是不太好写的。

posted @   feiko  阅读(1279)  评论(2编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
点击右上角即可分享
微信分享提示