浅谈牛顿迭代
前言
话说每次讲跟 n e w t o n newton newton 有关的东西,都得先 % \% % 一下这位大佬。
引入
我们很早就知道伽罗瓦证明了五次及以上方程不具有普遍的求根公式,但是高次方程却充斥着我们的生活,因此在不同的领域求解不同的高次方程变得尤为困难。
在高中我们就学过,对于在 [ l , r ] [l,r] [l,r] 上单调的函数 f ( x ) f(x) f(x),若 f ( l ) f(l) f(l) 与 f ( r ) f(r) f(r) 异号,即 f ( l ) ⋅ f ( r ) ≤ 0 f(l)\cdot f(r)\le0 f(l)⋅f(r)≤0 ,则在此区间内定有且仅有一个该函数 f ( x ) f(x) f(x) 的根。
结论很直观,这里就不赘述了。看到单调性,我们很容易想到可以通过二分的方法来快速找到函数在区间 [ l , r ] [l,r] [l,r] 内的根,这也是课本里教的方法。二分的复杂度已经是 Θ ( log 2 n ) \Theta(\log_2 n) Θ(log2n) 的了,可谓是极其优秀,但是随着时代的进步,这种 求近似根 的方法已经显得效率有点低了。
想必大家都知道 C + + C++ C++ 中自带的 S T L STL STL 函数 s q r t ( a ) sqrt(a) sqrt(a),它的作用是求 a a a 的算术平方根,我们也可以理解为求解一个 x 2 − a = 0 x^2-a=0 x2−a=0 的方程,或者说找到函数 f ( x ) = x 2 − a f(x)=x^2-a f(x)=x2−a 的正零点,亦可想到可以用我们学过的二分实现。
但是已经有大佬尝试过了,无论如何优化这种求近似根的二分,都无法超越 S T L STL STL 中自带的 s q r t sqrt sqrt 函数,甚至被远远甩开一大截!这是为什么?我曾经很不理解,知道我认识了它
n e w t o n m e t h o d newton~method newton method,牛顿迭代。
牛顿迭代
牛顿迭代也是一种求解方程近似根的算法,但是它的效率远超我们熟知的二分算法!
在了解牛顿迭代之前,我们必须得明白什么是导数。
导数
我们通常听到的导数定义是一个函数的瞬时变化率(瞬时不存在变化,因此这个“瞬时”没那么严谨,是一个极限概念而已,感性理解),但是在这里,对于 f ( x ) f(x) f(x) 的导数 f ′ ( x ) f'(x) f′(x) 最好的理解是, f ′ ( n ) f'(n) f′(n) 所表示的切线是函数 f ( n ) f(n) f(n) 当自变量为 n n n 时的线性近似
什么意思呢?就是说 f ′ ( x ) f'(x) f′(x) 是一条在 f ( x ) f(x) f(x) 图像上且过点 ( x , f ( x ) ) (x,f(x)) (x,f(x)) 的一条切线的斜率。
对,导数就是函数的切线的斜率函数。
考虑斜率的定义,函数上 x x x 点附近的切线的斜率就是 f ( x + Δ x ) − f ( x ) Δ x \dfrac{f(x+\Delta x)-f(x)}{\Delta x} Δxf(x+Δx)−f(x) , Δ x \Delta x Δx 是自变量 x x x 的增量,通常取它趋近于 0 0 0
举个例子最直观不过了,我们考虑对函数
f
(
x
)
=
x
2
f(x)=x^2
f(x)=x2 求导:
f
′
(
x
)
=
f
(
x
+
Δ
x
)
−
f
(
x
)
Δ
x
=
(
x
+
Δ
x
)
2
−
x
2
Δ
x
=
(
Δ
x
)
2
+
2
(
Δ
x
)
x
Δ
=
Δ
x
+
2
x
\begin{aligned} f'(x)&=\dfrac{f(x+\Delta x)-f(x)}{\Delta x}\\ &=\dfrac{(x+\Delta x)^2-x^2}{\Delta x}\\ &=\dfrac{(\Delta x)^2+2(\Delta x) x}{\Delta}\\ &=\Delta x+2x \end{aligned}
f′(x)=Δxf(x+Δx)−f(x)=Δx(x+Δx)2−x2=Δ(Δx)2+2(Δx)x=Δx+2x
再考虑导数的定义是极限的,因此我们对 Δ x \Delta x Δx 取极限
f ′ ( x ) = lim Δ x → 0 Δ x + 2 x = 2 x \begin{aligned} f'(x)&=\lim_{\Delta x\rightarrow 0}\Delta x+2x\\ &=2x \end{aligned} f′(x)=Δx→0limΔx+2x=2x
因此,
2
x
2x
2x 就是
f
(
x
)
=
x
2
f(x)=x^2
f(x)=x2 的导数。同理,考虑导数的几何意义,
2
t
(
x
−
t
)
+
t
2
2t(x-t)+t^2
2t(x−t)+t2 是一条切
x
2
x^2
x2 于
(
t
,
t
2
)
(t,t^2)
(t,t2) 点的切线。而切线则是在该点上函数的线性近似。(扣题)
知道了切线是函数的一种线性近似,自然会生出一种想法:我们可以通过找到切线的零点,然后转移到此处,再来作切线,再来得到零点……因为切线是原函数的近似,因此如此迭代往复,得到的点将平方逼近原函数的零点。
一不小心,你就发现了牛顿迭代的核心思想!
我们把过程量化,再来系统的讲一遍:
随便找到一个初始迭代的点 C C C- 得到在位置 C C C 处的函数切线,并找到这条切线的零点,作为下一个迭代点。
- 重复迭代步骤 2
大概就是这么个过程:
数学
牛顿迭代的整体思路真的很好理解,但是只有思想无法实现就没办法帮助我们写代码。因此,我们需要将牛顿迭代真正的公式化。
考虑此时我们的初始点为 x 0 x_0 x0,那么在该位置上的函数切线方程即为 y = f ′ ( x 0 ) ( x − x 0 ) + f ( x 0 ) y=f'(x_0)(x-x_0)+f(x_0) y=f′(x0)(x−x0)+f(x0),找到零点也就是等于求解一元一次方程 f ( x 0 ) ( x − x 0 ) + f ( x 0 ) = 0 f(x_0)(x-x_0)+f(x_0)=0 f(x0)(x−x0)+f(x0)=0
我们设 x 1 x_1 x1 为这个方程的解,也就是说 f ′ ( x 0 ) ( x 1 − x 0 ) + f ( x 0 ) = 0 f'(x_0)(x_1-x_0)+f(x_0)=0 f′(x0)(x1−x0)+f(x0)=0,化简可得 x 1 = x 0 − f ( x 0 ) f ′ ( x 0 ) x_1=x_0-\dfrac{f(x_0)}{f'(x_0)} x1=x0−f′(x0)f(x0)
此时,倘若我们继续以 x 1 x_1 x1 作为初始点如此迭代下去,得到的点将具有以下迭代关系:
x n + 1 = x n − f ( x n ) f ′ ( x n ) \Large x_{n+1}=x_n-\dfrac{f(x_n)}{f'(x_n)} xn+1=xn−f′(xn)f(xn)
显然, x n + 1 x_{n+1} xn+1 会快速接近零点。
使用这条公式,我们就可以写出牛顿迭代的代码:
#define exp 1e-10
double f(double x)
{
return /*函数表达式*/
}
double df(double x)
{
int dx=x+exp;
return 1.0*(f(dx)-f(x))/dx; //使用两个接近的点来模拟求导
}
double newton(double x0)
{
while(abs(f(x0))>1e-10) x0-=1.0*f(x0)/df(x0); //迭代公式,精度足够就退出
return x0;
}
应用
很多问题都可以转化为找函数零点,而这正是牛顿迭代所擅长的,这里举几个简单的例子:
- 求函数局部最优解
在函数类似山峰的部分,都有着导数为 0 0 0 的特性,因此找山峰实际上就是求导数的零点。 - 对函数求值
例如引入中的求 a \sqrt a a,我们可以理解为解 x 2 − a = 0 x^2-a=0 x2−a=0 的方程,依然是个找零点的过程。
更一般的,对于任意一个函数 f ( a ) f(a) f(a) 求值,其实际上就是在解 x = f ( a ) x=f(a) x=f(a) 的方程,化简得 f − 1 ( x ) − a = 0 f^{-1}(x)-a=0 f−1(x)−a=0,也是一个求零点的问题。
……
综上所述,在实际上,我们很多问题都需要用到牛顿迭代,而它的本质实际上感性上还是不难理解的。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· 没有源码,如何修改代码逻辑?
· PowerShell开发游戏 · 打蜜蜂
· 在鹅厂做java开发是什么体验
· WPF到Web的无缝过渡:英雄联盟客户端的OpenSilver迁移实战