zoukankan      html  css  js  c++  java
  • 迭代法

    hdu 3809;

    一,迭代法 的基本概念:

    迭代法事一种常用算法设计方法。迭代式一个不断用新值取代变量的旧值,或由旧值递推出变量的新值的过程。迭代机制需要以下一些要素:

    ①迭代表达式;

    ②迭代变量;

    ③迭代初值;

    ④迭代终止条件。

    当一个问题的求解过程能够由一个初值使用一个迭代表达式进行反复的迭代时,便可以用效率极高的重复程序描述,所以迭代也是用循环结构实现,只不过要重复的操作是不断从一个变量的旧值出发计算它的新值。其基本格式如下:

    迭代变量赋初值;

    循环语句

    {

    计算迭代式;

    新值取代旧值;

    }

    迭代一词在wiki百科 中的定义如下:

    迭代 是数值分析中通过从一个初始 估计 出发寻找一系列近似解 来解决问题(一般是解方程或者方程组)的过程,为实现这一过程所使用的方法统称为迭代法 (Iterative Method)。

    跟迭代法相对应的是直接法 (或者称为一次解法 ),即一次性解决问题,例如通过开方解决方程x2 = 4 。一般如果可能,直接解法总是优先考虑的。但当遇到复杂问题时,特别是在未知量很多,方程为非线性时,我们无法找到直接解法(例如五次以及更高次的代数方程没有解析解阿贝耳定理 ),这时候或许可以通过迭代法寻求方程(组)的近似解。 ,参见

    最常见的迭代法是牛顿法。其他还包括最速下降法 、共轭迭代法、变尺度迭代法 、最小二乘法、线性规划、非线性规划单纯型法惩罚函数法斜率投影法 、遗传算法、模拟退火等等。

    其中这里要指出一下的是 牛顿迭代法 ,是一种较为常用的迭代法。

    牛顿法 Newton's method )又称为牛顿-拉夫逊方法Newton-Raphson method ),它是一种在实数域和复数域上近似求解方程的方法。方法使用函数f (x ) 的泰勒级数的前面几项来寻找方程f (x ) = 0 的根。

    首先,选择一个接近函数f (x ) 零点的x0 ,计算相应的f (x 0 ) 和切线斜率f '(x0 ) (这里f ' 表示函数f的导数)。然后我们计算穿过点(x 0 ,f (x0 )) 并且斜率为f '(x 0 ) 的直线和x轴的交点的x 坐标,也就是求如下方程的解:

    x/cdot f'(x_0)+f(x_0)-x_0/cdot f'(x_0)=0

    我们将新求得的点的x 坐标命名为x1 ,通常x 1 会比x0 更接近方程f (x ) = 0 的解。因此我们现在可以利用x1 开始下一轮迭代。迭代公式可化简为如下所示:

    x_{n+1} = x_n - /frac{f(x_n)}{f'(x_n)}

    已经证明,如果f '连续 的,并且待求的零点x是孤立的,那么在零点x 周围存在一个区域,只要初始值x0 位于这个邻近区域内,那么牛顿法必定收敛。 并且,如果f '(x ) 不为0, 那么牛顿法将具有平方收敛的性能. 粗略的说,这意味着每迭代一次,牛顿法结果的有效数字将增加一倍。下图为一个牛顿法执行过程的例子。

    第一个例子

    求方程f (x ) = cos(x ) − x 3 的根。两边求导,得f  '(x ) = −sin(x ) − 3x2 。由于cos(x ) ≤ 1(对于所有x ),以及x 3 > 1(对于x >1),可知方程的根位于0和1之间。我们从x0 = 0.5开始。

    /begin{matrix}  x_1 & = & x_0 - /frac{f(x_0)}{f'(x_0)} & = & 0.5 - /frac{/cos(0.5) - 0.5^3}{-/sin(0.5) - 3 /times 0.5^2} & = & 1.112141637097 //  x_2 & = & x_1 - /frac{f(x_1)}{f'(x_1)} & = & /vdots & = & /underline{0.}909672693736 //  x_3 & = & /vdots & = & /vdots & = & /underline{0.86}7263818209 //  x_4 & = & /vdots & = & /vdots & = & /underline{0.86547}7135298 //  x_5 & = & /vdots & = & /vdots & = & /underline{0.8654740331}11 //  x_6 & = & /vdots &= & /vdots & = & /underline{0.865474033102}/end{matrix}

    第二个例子

    牛顿法亦可发挥与泰勒展开式,对于函式展开的功能。


    求a的m次方根。

    x m - a= 0

    f (x ) = x m a f '(x ) = m xm − 1

    而a的m次方根,亦是x的解,

    以牛顿法来迭代:

    x_{n+1} = x_n - /frac{f(x_n)}{f'(x_n)}

    x_{n+1} = x_n - /frac{x_n^m-a}{mx_n^{m-1}}

    x_{n+1} = x_n - /frac{x_n}{m}(1-ax_n^{-m})

    (或 x_{n+1} = x_n - /frac{1}{m}(x_n-a/frac{x_n}{x_n^m}) )

    另外,牛顿迭代法 百度词条 中的定义:点击这里

    二, 简单实例

    例题一:

    求Fibonacci数列。

    1,1,2,3,5,8,13,21,34..... 此为Fibonacci数列。

    输入:整数n(n>=3)

    输出:n个Fibonacci数列

    算法设计:基本迭代法

    f(1)=f(2)=1;

    f(n)=f(n-1)+f(n-2);    (n>=3)

    C++实现如下:


    #include<iostream>
    using namespace std;
    void f(int n)
    {
    	int f,f1,f2;
    	f1=f2=1;
    	cout<<f1<<" "<<f2<<" ";
    	for(int i=3;i<=n;i++)
    	{
    		f=f1+f2;
    		cout<<f<<" ";
    		f1=f2;
    		f2=f;
    	}
    	cout<<endl;
    }
    int main()
    {
    	int n;
    	cin>>n;
    	f(n);
    	return 0;
    }


    10
    1 1 2 3 5 8 13 21 34 55
    Press any key to continue

    例题二:

    编写一个程序,求以下方程的根:

    x³-5x²+6x-80=0

    算法设计:牛顿迭代法

    利用牛顿迭代公式 x(n+1)=x(n)-f(x(n))/f'(x(n));

    循环体为:

    for(int i=0;i<n;i++)
    {
            x1=x0-f(x0)/df(x0);
            x0=x1;
    }

    其中n为迭代次数,该值越大根值越精确;x0为在方程根取值范围内的任意值,作为初始迭代的条件。

    C++实现如下:

    #include<iostream>
    using namespace std;
    float f(float x)
    {
    	float y;
    	y=x*x*x-5.0*x*x+16.0*x-80.0;
    	return y;
    }
    float df(float x)
    {
    	float y;
    	y=3.0*x*x-10.0*x+6;
    	return y;
    }
    int main()
    {
    	float x0,x1;
    	int n;
    	cout<<"输入x0:";
    	cin>>x0;
    	cout<<"迭代次数:";
    	cin>>n;
    	for(int i=0;i<n;i++)
    	{
    		x1=x0-f(x0)/df(x0);
    		x0=x1;
    	}
    	cout<<"方程的根:x="<<x0<<endl;
    	return 0;
    }


    输入x0:2
    迭代次数:10
    方程的根:x=5.15874
    Press any key to continue

    输入x0:2
    迭代次数:100
    方程的根:x=5
    Press any key to continue



  • 相关阅读:
    【转】UTF8文件的Unicode签名BOM(Byte Order Mark)问题
    【转】java使用正则表达式去除字符串的html标签
    用UltraEdit转换大小写
    【转】浅谈字节序(Endianness)
    【转】用UltraEdit的正则表达式替换功能来格式化网页源代码
    【转】关于正则表达式匹配任意字符(包括换行符)的写法
    【转】C# 中的 #region 和 #endregion
    网易游戏开发工程师笔试题
    c++笔试题汇总
    恒生电子面试过程纪录
  • 原文地址:https://www.cnblogs.com/yyf573462811/p/6365370.html
Copyright © 2011-2022 走看看