This repository was archived by the owner on May 14, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathintegrator.cpp
More file actions
183 lines (150 loc) · 5.71 KB
/
integrator.cpp
File metadata and controls
183 lines (150 loc) · 5.71 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
#include <stdio.h>
#include <math.h>
#include "integrator.h"
#include "model.h"
// Определение машинного нуля
template <class T>
struct STypeZero
{
STypeZero()
{
T l_tTmp = 1;
while ( ( l_tTmp /= 2 ) + 1 > 1 )
value = l_tTmp;
}
T value;
};
STypeZero< long double > g_ldZero;
#define max(a, b) ( ( (a) > (b) ) ? (a) : (b) )
#define min(a, b) ( ( (a) < (b) ) ? (a) : (b) )
//===========================================================================
// class TDormandPrinceIntegrator
const long double TDormandPrinceIntegrator::c[7] = { 0, 1./5, 3./10, 4./5, 8./9, 1., 1. };
const long double TDormandPrinceIntegrator::a[7][6] = {
{ 0. },
{ 1./5 },
{ 3./40, 9./40 },
{ 44./45, -56./15, 32./9 },
{ 19372./6561, -25360./2187, 64448./6561, -212./729 },
{ 9017./3168, -355./33, 46732./5247, 49./176, -5103./18656 },
{ 35./384, 0., 500./1113, 125./192, -2187./6784, 11./84 }
};
const long double TDormandPrinceIntegrator::b1[7] = { 35./384, 0., 500./1113, 125./192, -2187./6784, 11./84, 0 };
const long double TDormandPrinceIntegrator::b2[7] = { 5179./57600, 0., 7571./16695, 393./640, -92097./339200, 187./2100, 1./40 };
//---------------------------------------------------------------------------
TDormandPrinceIntegrator::TDormandPrinceIntegrator()
: TIntegrator()
{
// Определение машинного нуля
double v = 1.;
while (1.+v > 1.) {
u = v;
v = v/2.;
}
}
//---------------------------------------------------------------------------
long double TDormandPrinceIntegrator::Run(TModel* Model)
{
//printf ("\n1.1");
long double // Это время для интегрирования (увеличивается на величину шага интегрирования)
t = Model->getT0(),
// Это время для выдачи (увеличивается дискретно на величину плотности)
t_out = t,
// Это конечное время
t1 = Model->getT1(),
// Это шаг интегрирования
h,
// Это шаг после коррекции (инициализируется плотностью выдачи результата)
h_new = Model->getSamplingIncrement(),
// Это ошибка на шаге интегрирования
e = 0;
TVector // Это вектор состояния на начало шага
X = Model->getInitialConditions(),
// Это вектор состояния на конец шага (решение 4-го порядка)
X1( X.size() ),
// Это вектор состояния на конец шага для коррекции величины шага (решение 5-го порядка)
X2( X.size() ),
// Это вектор для выдачи результата
Xout ( X.size() ),
// Это буфер для вычисления коэффициентов К
Y( X.size() );
// Подготовка хранилища результатов в модели для повышения эффективности выделения памяти
Model->prepareResult();
// Инициализируем размерности коэффициентов K[j]
for ( int j = 7; j > 0; --j, K[j].resize( X.size() ) );
//printf ("\n1.2");
// Счётчик количества сделанных шагов
int N = 0;
//printf ("\n&&&&&&K\n");
// Главный цикл
while ( t < t1 )
{
// Устанавливаем шаг на итерацию
h = h_new;
// Вычисляем коэффициенты К
for ( int j = 0; j < 7; j++ )
{
for ( int k = X.high(); k >= 0; k-- )
{
Y[k] = X[k];
for ( int s = 0; s < j; s++ )
{
Y[k] += K[s][k] * a[j][s] * h;
}
}
Model->getRight( Y, t + c[j] * h, K[j] );
}
// Вычисляем новые значения результатов и локальной ошибки (4 и 5 порядок)
e = 0;
for ( int k = X.high(); k >= 0; k-- )
{
X1[k] = X2[k] = X[k];
for ( int j = 0; j < 7; j++ )
{
X1[k] += K[j][k] * b1[j] * h;
X2[k] += K[j][k] * b2[j] * h;
}
e += powl( h * (X1[k] - X2[k]) / max( max( fabsl(X[k]), fabsl(X1[k]) ), max((long double)1e-5, 2*g_ldZero.value/Eps) ) , 2 );
}
e = sqrtl( e / X.size() );
// Коррекция шага
h_new = h / max( 0.1, min( 5., powl(e / Eps, 0.2)/0.9 ) );
// Если локальная ошибка превышает установленную величину, пытаемся сделать шаг заново
if ( e > Eps )
continue;
// Формирование результатов при помощи механизма плотной выдачи
while ( (t_out < t + h) && (t_out <= t1) )
{
long double l_ldTheta = (t_out - t)/h,
b[6];
// Рассчитываем коэффициенты плотной выдачи
b[0] = l_ldTheta * ( 1 + l_ldTheta*(-1337./480. + l_ldTheta*(1039./360. + l_ldTheta*(-1163./1152.))));
b[1] = 0;
b[2] = 100. * powl(l_ldTheta, 2) * (1054./9275. + l_ldTheta*(-4682./27825. + l_ldTheta*(379./5565.)))/3.;
b[3] = -5. * powl(l_ldTheta, 2) * (27./40. + l_ldTheta*(-9./5. + l_ldTheta*(83./96.)))/2.;
b[4] = 18225. * powl(l_ldTheta, 2) * (-3./250. + l_ldTheta*(22./375. + l_ldTheta*(-37./600.)))/848.;
b[5] = -22. * powl(l_ldTheta, 2) * (-3./10. + l_ldTheta*(29./30. + l_ldTheta*(-17./24.)))/7.;
// Получаем результат для выдачи
for ( int k = X.high(); k >= 0; k-- )
{
long double l_ldSum = 0;
for ( int j = 5; j >= 0; j-- )
l_ldSum += b[j] * K[j][k];
Xout[k] = X[k] + h * l_ldSum;
}
// Передача результата в модель
Model->addResult( Xout, t_out );
// Наращиваем время выдачи
t_out += Model->getSamplingIncrement();
}
// Обновляем X и наращиваем время на величину сделанного шага
X = X1;
t += h;
// Считаем количество итераций для вычисления глобальной погрешности
N++;
}
//printf ("\n1.4");
// printf ("\n%i", N);
// Возвращаем величину глобальной погрешности
return Eps / pow( N, 1.5 );
}