
[导读] 前面刚转了一篇文章提到了牛顿-拉夫逊(拉弗森)(Newton-Raphson method)方法,感觉这个数学方法很有必要相对深入写一篇文章来总结分享印证一下自己的理解。这是写本文的由来,如果发现文章中有错误之处,请留言交流讨论。! _1 e% g( z" f# V+ Q7 j 0 p E4 I4 s; \ ![]()
![]() 所以,牛顿迭代法(简写)就是一种近似求解实数域与复数域求解方程的数学方法。那么这个方法是具体是什么原理呢?: F6 f V, {, L" j2 a% s3 s/ Y 直接看数学公式描述如何迭代不直观,先来看动图就很容易理解牛顿迭代法为什么叫迭代法以及怎样迭代的: 牛顿迭代法是原理是根据一个初始点在该点做切线,切线与X轴相交得出下一个迭代点的坐标,再在处做切线,依次类推,直到求得满足精度的近似解为止。
由前面描述知道,牛顿迭代法是用来近似求解方程的,这里有两个点需要说明:
来看看,数学上如何描述的?
来简单推一推上面公式的由来,直线函数方程为:
如何编码呢? 由于牛顿迭代法主要目的是解方程,当然也有可能用于某一个数学函数求极值,所以无法写出通用的代码,这里仅仅给出一个编代码的思路。相信掌握了思路,对于各种实际应用应该能很快的写出符合实际应用的代码。 假定一函数为
其波形图如下: ![]() 其一阶导数为:
那么对于该函数的根:
从图上大致可以知道有两个根,如果直接解方程,则很难求出其根,可以编个代码试试: #include <stdio.h>" h5 ~) }& G+ \/ ?' m- k#include <math.h> #include <stdlib.h> /*假定待求根函数如下*/ #define F(x) (2*(x)*(x)-10*cos(x)+(x)-80) ! t0 F% t7 T! k" t /*其一阶导数为*/ #define DF(x) (4*(x)+10*sin(x)+1). ~# N% u6 s7 X& d3 b4 A 4 {$ Q: Q/ L4 f+ ~; `, Q0 v float newton_rooting(float x0,float precision,float min_deltax,int max_iterations)5 k. e7 P; k9 j4 {3 Y$ J! } { float xn,xn1,fn,fn1,dfn; float deltax;* M. R' h4 f* I* I0 A int step = 0; xn = x0; xn1 = x0; do{ xn = xn1; fn = F(xn);+ U6 ]' H% u8 Q7 | dfn = DF(xn); /*判0*/ if( fabs(dfn) <1e-6 ) { if( fabs(fn)>precision ) return NAN; else return fn;0 @$ H! H3 N7 k) F# v } xn1 = xn - fn/dfn; fn1 = F(xn1);( d8 z1 f! Q5 s+ D2 I! F& W- B deltax = fabs(xn1-xn);: H. K# L: @) E step++; if( step>max_iterations )+ E2 H8 a% D( G) p5 o; [ { if( fabs(fn1)<precision ): Y( \: J0 r" e, k# n return xn;* q+ k7 \, Y7 ]: h4 j else return NAN; }: { X. h* i5 I M: o5 ] ] } while( fabs(fn1)>precision || deltax>min_deltax ); return xn1;- k7 S2 s. s7 |. C8 s, [. ~0 i }% S3 b1 D% W$ n" R! ` , G. F- S" Z8 _/ ]7 |; C void main() {) ~: e' @0 O) C# G3 j1 T/ } float root_guess = 23.0f;( n( x% x: E, U) u* ]0 @ float precision = 0.00001f; float min_deltax = 0.001f; float root;( N" f8 V1 R( u% ~) ]( h% O int step = 7; ! ?. E) ?9 u7 Z( f ], B. k root = newton_rooting( root_guess,precision,min_deltax,step );) \4 l0 t9 L, e! S4 w% y/ S1 P printf("根为: %f,函数值为:%f\n", root,F(root)); root_guess = -23;# I9 l) K+ L9 n/ w; r/ {: T root = newton_rooting( root_guess,precision,min_deltax,step );& D+ E* r) ^/ l" ~/ Q printf("根为: %f,函数值为:%f\n", root,F(root)); u; A- _: |, p, H% P7 Z+ ]; u } 结果: 根为: 6.457232, 函数值为:0.000004根为: -6.894969,函数值为:-0.000008 函数值已经很接近于0了,如果还需要更为精确的值,则可以选择在此基础上进一步求解,比如利用二分法逼近。 需要注意些啥?
![]()
![]()
所以实际应用时,需要知道自己待求解模型的大致情况,在合理的加以调整。 有哪些应用?
牛顿迭代法在解决实际问题时,利用迭代求方程近似根的数学原理,在工程中有着很好的实用价值。比如求一个趋势的极值,传递函数参数辨识等都有广泛的实际应用。本文抛砖引玉,有可能文章也有很多错误疏漏的地方,如有不同看法或者发现错误,欢迎留言交流指正。 ~4 ]( |: i B. d" K7 ?6 v) h+ H `. e, J! `4 j |
大大的点赞,好厉害! |