27namespace ButcherTableau
30#if defined(__GNUC__) && !defined(__clang__)
31 #pragma GCC diagnostic push
32 #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
43template<
typename Y,
typename Z, std::
floating_po
int Scalar,
size_t s, std::array<std::array<Scalar, s + 1>, s + 1> butcherTableau>
44inline Y
RungeKuttaExplicit(
const Y& y_n,
const std::array<Z, s>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n)
46 static_assert(gcem::abs(
static_cast<Scalar
>(1.0) - std::accumulate(butcherTableau[s].begin(), butcherTableau[s].end(),
static_cast<Scalar
>(0.0))) < 1e-8,
47 "The sum of the last row in the Butcher tableau has to be 1");
48 for (
size_t r = 0; r <= s; ++r)
50 for (
size_t c = r + 1; c <= s; ++c)
52 INS_ASSERT_USER_ERROR(butcherTableau.at(r).at(c) == 0.0,
"All terms in the upper triangle have to be 0");
62 for (
size_t i = 1; i <= s; ++i)
64 auto b = butcherTableau[s].at(i);
65 auto c = butcherTableau.at(i - 1)[0];
69 for (
size_t j = 2; j <= i; ++j)
71 auto a = butcherTableau.at(i - 1).at(j - 1);
72 if (j == 2) { sum_a = a * k.at(j - 2); }
73 else { sum_a += a * k.at(j - 2); }
82 if (i < 2) { k.at(i - 1) = f(y_n, z.at(i - 1), constParam, t_n + c * h); }
83 else { k.at(i - 1) = f(y_n + sum_a * h, z.at(i - 1), constParam, t_n + c * h); }
89 if (i == 1) { sum_b = b * k.at(i - 1); }
90 else { sum_b += b * k.at(i - 1); }
102 return y_n + h * sum_b;
105#if defined(__GNUC__) && !defined(__clang__)
106 #pragma GCC diagnostic pop
110template<
typename Scalar>
111constexpr std::array<std::array<Scalar, 2>, 2>
RK1 = { { { 0.0, },
116template<
typename Scalar>
117constexpr std::array<std::array<Scalar, 3>, 3>
RK2 = { { { 0.0, },
120 { 0.0, 0.0, 1.0 } } };
123template<
typename Scalar>
124constexpr std::array<std::array<Scalar, 3>, 3>
Heun2 = { { { 0.0, },
127 { 0.0, 0.5, 0.5 } } };
130template<
typename Scalar>
131constexpr std::array<std::array<Scalar, 4>, 4>
RK3 = { { { 0.0, },
135 { 0.0, 1.0 / 6.0, 4.0 / 6.0, 1.0 / 6.0 } } };
138template<
typename Scalar>
139constexpr std::array<std::array<Scalar, 4>, 4>
Heun3 = { { { 0.0, },
140 { 1.0 / 3.0, 1.0 / 3.0 },
141 { 2.0 / 3.0, 0.0, 2.0 / 3.0 },
143 { 0.0, 1.0 / 4.0, 0.0, 3.0 / 4.0 } } };
146template<
typename Scalar>
147constexpr std::array<std::array<Scalar, 5>, 5>
RK4 = { { { 0.0, },
150 { 1.0, 0.0, 0.0, 1.0 },
152 { 0.0, 1.0 / 6.0, 1.0 / 3.0, 1.0 / 3.0, 1.0 / 6.0 } } };
174template<
typename Y,
typename Z, std::
floating_po
int Scalar>
175Y
RungeKutta1(
const Y& y_n,
const std::array<Z, 1>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
177 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 1, ButcherTableau::RK1<Scalar>>(y_n, z, h, f, constParam, t_n);
203template<
typename Y,
typename Z, std::
floating_po
int Scalar>
204Y
RungeKutta2(
const Y& y_n,
const std::array<Z, 2>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
206 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 2, ButcherTableau::RK2<Scalar>>(y_n, z, h, f, constParam, t_n);
232template<
typename Y,
typename Z, std::
floating_po
int Scalar>
233Y Heun2(
const Y& y_n,
const std::array<Z, 2>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
235 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 2, ButcherTableau::Heun2<Scalar>>(y_n, z, h, f, constParam, t_n);
263template<
typename Y,
typename Z, std::
floating_po
int Scalar>
264Y
RungeKutta3(
const Y& y_n,
const std::array<Z, 3>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
266 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 3, ButcherTableau::RK3<Scalar>>(y_n, z, h, f, constParam, t_n);
294template<
typename Y,
typename Z, std::
floating_po
int Scalar>
295Y Heun3(
const Y& y_n,
const std::array<Z, 3>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
297 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 3, ButcherTableau::Heun3<Scalar>>(y_n, z, h, f, constParam, t_n);
327template<
typename Y,
typename Z, std::
floating_po
int Scalar>
328Y
RungeKutta4(
const Y& y_n,
const std::array<Z, 4>& z,
const Scalar& h,
const auto& f,
const auto& constParam,
const Scalar& t_n = 0)
330 return ButcherTableau::RungeKuttaExplicit<Y, Z, Scalar, 4, ButcherTableau::RK4<Scalar>>(y_n, z, h, f, constParam, t_n);
#define INS_ASSERT_USER_ERROR(_EXP, _MSG)
Assert function with message.
Definition Assert.h:21
Utility class for logging to console and file.
constexpr std::array< std::array< Scalar, 5 >, 5 > RK4
Butcher tableau for Runge-Kutta 4th order (explicit)
Definition NumericalIntegration.hpp:147
Y RungeKutta3(const Y &y_n, const std::array< Z, 3 > &z, const Scalar &h, const auto &f, const auto &constParam, const Scalar &t_n=0)
Runge-Kutta 3rd order (explicit) / Simpson's rule .
Definition NumericalIntegration.hpp:264
Y RungeKuttaExplicit(const Y &y_n, const std::array< Z, s > &z, const Scalar &h, const auto &f, const auto &constParam, const Scalar &t_n)
Calculates explicit Runge-Kutta methods. Order is defined by the Butcher tableau.
Definition NumericalIntegration.hpp:44
constexpr std::array< std::array< Scalar, 4 >, 4 > Heun3
Butcher tableau for Heun's method (3nd order) (explicit)
Definition NumericalIntegration.hpp:139
Y RungeKutta1(const Y &y_n, const std::array< Z, 1 > &z, const Scalar &h, const auto &f, const auto &constParam, const Scalar &t_n=0)
Runge-Kutta 1st order (explicit) / (Forward) Euler method .
Definition NumericalIntegration.hpp:175
constexpr std::array< std::array< Scalar, 3 >, 3 > Heun2
Butcher tableau for Heun's method (2nd order) (explicit)
Definition NumericalIntegration.hpp:124
Y RungeKutta4(const Y &y_n, const std::array< Z, 4 > &z, const Scalar &h, const auto &f, const auto &constParam, const Scalar &t_n=0)
Runge-Kutta 4th order (explicit) .
Definition NumericalIntegration.hpp:328
Y RungeKutta2(const Y &y_n, const std::array< Z, 2 > &z, const Scalar &h, const auto &f, const auto &constParam, const Scalar &t_n=0)
Runge-Kutta 2nd order (explicit) / Explicit midpoint method .
Definition NumericalIntegration.hpp:204
constexpr std::array< std::array< Scalar, 4 >, 4 > RK3
Butcher tableau for Runge-Kutta 3rd order (explicit) / Simpson's rule.
Definition NumericalIntegration.hpp:131
constexpr std::array< std::array< Scalar, 3 >, 3 > RK2
Butcher tableau for Runge-Kutta 2nd order (explicit) / Explicit midpoint method.
Definition NumericalIntegration.hpp:117
constexpr std::array< std::array< Scalar, 2 >, 2 > RK1
Butcher tableau for Runge-Kutta 1st order (explicit) / (Forward) Euler method.
Definition NumericalIntegration.hpp:111