综述:
????????在一阶常微分方程的数值解法中 ,常采用的欧拉方法,欧拉方法局部截断误差为O(2),所以欧拉方法是一阶方法,改进后的欧拉方法截断误差为O(3),所以改进后的欧拉方法是二阶方法,精度更高。本文将以具体一微分方程为例,使用C++分别采用欧拉方法和改进后的欧拉方法对其求解。
目录
综述:
题目:
求解:
一、欧拉方法
二、改进欧拉方法
代码实现:
运行结果:
计算结果比较:
题目:
求初值问题:
data:image/s3,"s3://crabby-images/5cc1a/5cc1a64e43e560e6de53ada6e8acb4f6aca5cd23" alt="\left\{ \begin{array}{lr} y'=y-\frac{2x}{y} \quad,0\leqslant x\leqslant 1 & \\ y(0)=1 \end{array} \right."
求解:
一、欧拉方法
step1:
利用差商 代替 ?得
data:image/s3,"s3://crabby-images/82e04/82e043551a8232f0477b6cdc80126f2f31e54c15" alt="y(x_{n+1})\approx f(x_n,y(x_n))\quad (n=0,1,2,...)"
step2:
用 表示 的近似值,用 表示 的近似值,变为:
data:image/s3,"s3://crabby-images/07e5c/07e5c75252149061fe0648bc25db5641e987a659" alt="\left\{ \begin{array}{lr} y_{n+1}=y_n+hf(x_n,y_n) \quad (n=0,1,...)\\ y(0)=y(a)\end{array} \right."
二、改进欧拉方法
step1:
先用显式欧拉公式作预测,算出data:image/s3,"s3://crabby-images/74cbb/74cbba308d9911826a8dfae2850342eadf94bdda" alt="\bar{y}_{n+1}=y_n+hf(x_n,y_n)"
step2:
再将 代入隐式梯形公式的右边做校正得到
![\bar{y}_{n+1}=y_n+\frac{h}{2}[f(x_n,y_n)+f(x_{n+1},\bar{y}_{n+1})]](https://latex.codecogs.com/gif.latex?%5Cbar%7By%7D_%7Bn+1%7D%3Dy_n+%5Cfrac%7Bh%7D%7B2%7D%5Bf%28x_n%2Cy_n%29+f%28x_%7Bn+1%7D%2C%5Cbar%7By%7D_%7Bn+1%7D%29%5D)
?为了方便编程计算,我们还可以将改进的欧拉公式写成:
data:image/s3,"s3://crabby-images/454b1/454b10f29f51a814a1ffee96ff7085fd7edb100f" alt="y_p=y_n+hf(x_n,y_n)"
data:image/s3,"s3://crabby-images/37688/37688c4946bba6c6a5f2970f053482abd1250940" alt="y_c=y_n+hf(x_{n+1},y_p)"
data:image/s3,"s3://crabby-images/0ba04/0ba04f8ff0d1ffe13d4c628f20b99dddb8d78303" alt="y_{n+1}=\frac{1}{2}(y_p+y_c)"
代码实现:
公式比较简单,代码实现起来也很简单
#include <iostream>
#include <iomanip>
using namespace std;
class Euler
{
public:
/*Euler()
{
}*/
Euler(double x0,double y0,double h,int n)
{
this->h = h;
this->x = x0;
this->y = y0;
this->m_n = n;
}
double fun(double x, double y)
{
return (y - (x * 2.0) / y);
}
void Eulerfun()
{
double yn_next = 0.0;
int n = this->m_n;
while (n)
{
yn_next = this->y + this->h * fun(this->x, this->y);
cout <<"n = "<<this->m_n-n+1 << "时:\t" << "yn = " << setprecision(7) << yn_next << endl;
this->x += this->h;
this->y = yn_next;
n--;
}
}
double x;
double y;
double h;
int m_n;
};
class EulerPro :public Euler
{
public:
/*EulerPro():Euler(0, 1, 0.1, 10)
{
}*/
EulerPro(double x0, double y0, double h, int n):Euler(0,0,0,0)
{
this->h = h;
this->x = x0;
this->y = y0;
this->m_n = n;
}
void EulerProfun()
{
double yn_next, yp, yc;
yn_next = yp = yc = 0.0;
int n = this->m_n;
while (n)
{
yp = this->y + this->h * fun(this->x, this->y);
yc = this->y + this->h * fun(this->x + h, yp);
yn_next = (yp + yc) / 2;
cout << "n = " << this->m_n - n + 1 << "时:\t" << "yn = "<<yn_next << endl;
this->x += this->h;
this->y = yn_next;
n--;
}
}
};
int main()
{
cout << "--------------欧拉方法--------------" << endl;
cout << endl;
Euler e(0, 1, 0.1, 10);
e.Eulerfun();
cout << endl;
cout << "--------------改进欧拉--------------" << endl;
cout << endl;
EulerPro ep(0, 1, 0.1, 10);
ep.EulerProfun();
return 0;
}
运行结果:
data:image/s3,"s3://crabby-images/a1fa9/a1fa92b668fce84911a4a062c2e0f4a2d7f57a77" alt=""
计算结果比较:
n | data:image/s3,"s3://crabby-images/52d8d/52d8dc04cf324043cbd0e95d0fb9019b7a7b1c61" alt="x_n" | 欧拉方法data:image/s3,"s3://crabby-images/5276d/5276d6899a78528a45cc5c3e283bf0656c5959ba" alt="y_n" | 改进的欧拉方法data:image/s3,"s3://crabby-images/5276d/5276d6899a78528a45cc5c3e283bf0656c5959ba" alt="y_n" | 精确解data:image/s3,"s3://crabby-images/342b7/342b70aecd2b417e7ffdba80bdb2441c82eda67c" alt="y(x_n)" | 0 | 0 | 1 | 1 | 1 | 1 | 0.1 | 1.1 | 1.095909 | 1.095445 | 2 | 0.2 | 1.191818 | ?1.184097 | 1.183216 | 3 | 0.3 | 1.277438 | 1.266201 | 1.264991 | 4 | 0.4 | ?1.358213 | 1.34336 | 1.341641 | 5 | 0.5 | 1.435133 | 1.416402 | 1.414214 | 6 | 0.6 | ?1.508966 | 1.485956 | 1.483240 | 7 | 0.7 | 1.580338 | ?1.552514 | 1.549193 | 8 | 0.8 | ?1.649783 | 1.616475 | 1.612452 | 9 | 0.9 | 1.717779 | ?1.678166 | 1.673320 | 0 | 1 | 1.784771 | 1.737867 | 1.763051 |
明显看出,改进的欧拉方法算出的结果更接近真实结果,并且,改进的欧拉方法比欧拉方法的精度提升了一位。
|