Explicit methods#
Explicit solver methods take information up to the current solution in order to predict the next step.
Runge-Kutta methods#
The Runge-Kutta methods are a class of time stepping techniques where the next time step is calculated from the current time step and an estimate of its slope (rate of change). Increased accuracy is achieved through approximating higher order derivatives to improve our timestep. Since we don’t generally have that information, we can sample the function at different steps to approximate them. This is the basis for the Runge-Kutta family of methods.
Consider the Taylor expansion, $\(\begin{align} y_{i+1} &= y_i + y_i^\prime h + \frac{y_i^{\prime\prime}}{2}h^2 + \frac{y'''}{6}h^3 + \cdots \\ &= y_i + f(x_i,y_i) h + \frac{f(x_i,y_i)^\prime}{2}h^2 + \frac{f(x_i,y_i)^{\prime\prime}}{6}h^3 + \cdots \\ &\approx y_i + h \sum_{n=1}^s a_n k_n \end{align}\)$
where the last line is the general form of the Runge-Kutta methods. \(a_n\) are a set of constants and \(k_n\) are the function evaluated at different positions in the interval. The goal is clearly to match $\(\sum_{n=1}^s a_n k_n \approx f(x_i,y_i) + \frac{f(x_i,y_i)^\prime}{2}h + \frac{f(x_i,y_i)^{\prime\prime}}{6}h^2 + \cdots \)$
in so far as possible with a series truncated in \(s\) terms (called stages).
Let’s build up to the general form step by step.
Forward Euler method#
The Forward Euler method (aka: Euler-Cauchy / point-slope method, Explicit Euler) is the simplest Runge-Kutta method. One simply takes the slope at the current point, which is given in the differential equation, and assumes it is constant over the step:
Given, $\( \frac{dy}{dx} = f(x, y) \)\( the next time step is: \)\( y_{i+1} = y_i + f(x_i, y_i) h\)$
which is simply the left Riemann sum, and similarly the error is \(O(h^2)\) over the step, and \(O(h)\) over the full solution.
Heun’s method#
Now that we have a value for \(y_{i+1}\) is there a way we can use the prediction to correct itself? Heun’s method does exactly this and is called a predictor-corrector algorithm as a result.
Consider:
make a prediction, $\( y^0_{i+1} = y_i + f(x_i, y_i) h\)\( Now take the average of the slopes: \)\(\begin{align} \bar{y}' &= \frac{y_i+y_{i+1}}{2} \\ &= \frac{f(x_i,y_i)+f(x_{i+1}, y^0_{i+1})}{2} \end{align} \)\( which is used in the corrector: \)\( y_{i+1} = y_i + \frac{f(x_i,y_i)+f(x_{i+1}, y^0_{i+1})}{2} h\)$
This is interesting since one could repeat the predictor-corrector cycle with the intention of converging towards the correct answer, but we will see this is a bad idea:
Example: Heun’s method with multiple predictor-corrector cycles#
Integrate $\(y' = 4 e^{0.8 x}-0.5 y\)\( with \)\( y(0)=2\)$ using Heun’s method with 1 and 15 predictor-corrector cycles.
import numpy as np
def f(x, y):
return 4 * (2.71828 ** (0.8 * x)) - 0.5 * y
def heuns_method(x0, y0, h, t, iterations):
"""
Applies Heun's method to approximate the solution of a differential equation.
Args:
x0: The initial x value.
y0: The initial y value.
h: The step size.
t: The target time for the approximation.
iterations: The number of iterations for the predictor-corrector cycle.
Returns:
The approximate y value at time t.
"""
x = x0
y = y0
while x < t:
y_pred = y + h * f(x, y)
y_next = y + h * (f(x, y) + f(x + h, y_pred)) / 2
for _ in range(iterations - 1):
y_pred = y_next
y_next = y + h * (f(x, y) + f(x + h, y_pred)) / 2
y = y_next
x += h
print("Time ", x, ", approximation, ", y_next, ', True ,', 4/1.3*(np.exp(.8*x)-np.exp(-.5*x))+2*np.exp(-.5*x))
return y
# Initial conditions
x0 = 0
y0 = 2
# Step size and target time
h = 1
t = 4
# Estimate integral with Heun's method, iterating once
print("Heun's method with 1 iteration")
y_approx_once = heuns_method(x0, y0, h, t, 1)
# Estimate integral with Heun's method, iterating 15 times
print("\nHeun's method with 15 iterations")
y_approx_15 = heuns_method(x0, y0, h, t, 15)
Heun's method with 1 iteration
Time 1 , approximation, 6.701079461759733 , True , 6.194631377209372
Time 2 , approximation, 16.319768581929353 , True , 14.84392190764649
Time 3 , approximation, 37.199199627848756 , True , 33.67717176796817
Time 4 , approximation, 83.33761313495894 , True , 75.33896260915857
Heun's method with 15 iterations
Time 1 , approximation, 6.360863570675189 , True , 6.194631377209372
Time 2 , approximation, 15.302225064771143 , True , 14.84392190764649
Time 3 , approximation, 34.74323213193692 , True , 33.67717176796817
Time 4 , approximation, 77.73495685652544 , True , 75.33896260915857
Unfortunatley, Heun’s method does converge but not necessarily to the correct answer!
The Trapezoid rule#
In the case that the slope doesn’t depend on the function value, \(y'(x) = f(x)\) the predictor can be calcualted directly, eliminating the cycle:
which is mearly the trapezoid rule which carries \(O(h^3)\) accuracy locally and \(O(h^2)\) globally.
The Midpoint method#
Recall from the discussion on on differentiation / integration that information from the midpoint was often superior that of either endpoint (in isolation) as over/under estimates tend to cancel.
As an alternative to Heun’s predictor-corrector method, let’s subdivide the interval and find the slope at the midpoint. To do this, take a half step:
then use the slope at the midpoint, \(y'_{i+1/2} = f(x_{i+1/2},y_{i+1/2})\) to estimate the full step:
which has \(O(h^3)\) local / \(O(h^2)\) global error.
Explicit Runge-Kutta methods#
We are now in a position to generalize the RK methods. Recalling:
For the explicit set of RK methods,
The \(p_n\) and \(q_{nm}\), along with \(a_n\) are constants and determine the type of RK method.
Note RK-1 is simply the Forward Euler equation.
Butcher Tableaus#
Noting that \(a_n\), \(p_n\), and \(q_{nm}\) are 2 vectors of dimension \(s\) and an \(s \times s\) matrix, RK schemes can be written compactly in a Butcher Tableau. For explicit methods, this looks like:
Note that the \(q_{nm}\) matrix is lower triangular. We will soon introduce the implicit family of RK methods which are upper triangular!
RK-2 methods#
For a given order, the values of \(a_n\), \(p_n\), and \(q_{nm}\) are derived such that
The second order RK method is:
with $\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + p_2 h, y_i+q_{21} k_1 h) \end{align} \)$
Taylor expanding \(k_2\) in \(p\) and \(q\), $\( f(x_i + p_2 h, y_i+q_{21} k_1 h) = f(x_i,y_i) + p_2 h \frac{\partial f}{\partial x} + q_{21} k_1 h \frac{\partial f}{\partial y} + O(h^2)\)$
which plugged back in to \(y_{i+1}\): $\(y_{i+1} = y_i + [a_1+a_2] f(x_i, y_i) h + \bigg[a_2 p_2 \frac{\partial f}{\partial x} + a_2 q_{21} f(x_i, y_i) \frac{\partial f}{\partial y} \bigg] h^2 + O(h^3) \)$
which we can compare to a second order Taylor expansion: $\( \begin{align} y_{i+1} &= y_i + f(x_1, y_1) h + \frac{f'(x_1, y_1)}{2} h^2 \\ &= y_i + f(x_1, y_1) h + \bigg[\frac{\partial f}{\partial x} +\frac{\partial f}{\partial x} \frac{dy}{dx} \bigg] \frac{h}{2} \end{align}\)$
Comparing terms we see: $\(\begin{align} a_1+a_2 &= 1 \\ a_2 p_2 &= \frac{1}{2} \\ a_2 q_{21}&= \frac{1}{2}\\ \end{align} \)$
Here we see that we have a single degree of freedom for the set of constants! Any choice will satisfy 2’nd order equations and therefor ebe exact for constant, linear, or quadratic ODEs. Certaint choices will have better properties in general.
Heun’s method#
With \(a_2 = \frac{1}{2}\),
with $\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + h, y_i + k_1 h) \end{align} \)$
which is simply Heun’s method.
\begin{array}{c|cc} 0 & & \ 1 & 1 & \ \hline & \frac{1}{2} & \frac{1}{2} \ \end{array}
Midpoint method#
With \(a_2 = 1\), $\(y_{i+1} = y_i + k_2 h \)\( with \)\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + \frac{1}{2}h, y_i + \frac{1}{2} k_1 h) \end{align} \)$ which is the midpoint method.
\begin{array}{c|cc} 0 & & \ \frac{1}{2} & \frac{1}{2} & \ \hline & 0 & 1 \ \end{array}
Ralston’s method#
The choice \(a_2 = \frac{2}{3}\), can be shown to provide a minimum bound on the truncation error, $\(y_{i+1} = y_i + [k_1 + 2 k_2 ] \frac{h}{3} \)\( with \)\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + \frac{3}{4}h, y_i + \frac{3}{4} k_1 h) \end{align} \)$
\begin{array}{c|cc} 0 & & \ \frac{2}{3} & \frac{2}{3} & \ \hline & \frac{1}{4} & \frac{3}{4} \ \end{array}
RK-3 methods#
A similar derivation follows for RK3, again with choices for the missing degrees of freedom. A common choice is:
with $\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + \frac{1}{2}h, y_i + \frac{1}{2} k_1 h) \\ k_3 &= f(x_i + h, y_i - k_1 h + 2 k_2 h) \\ \end{align} \)$
which reduces to Simpson’s 1/3 Rule if \(f\) is only a function of \(x\). As with Simpson’s rule, it is \(O(h^4)\) local / \(O(h^3)\) global error.
The Butcher Tableau is: \begin{array}{c|ccc} 0 & & & \ \frac{1}{2} & \frac{1}{2} & & \ 1 & -1 & 2 & \ \hline & \frac{1}{6} & \frac{2}{3} & \frac{1}{6} \ \end{array}
RK4#
RK4 is the most common implementation. The ‘Classical Runge-Kutta method’ is:
with $\(\begin{align} k_1 &= f(x_i, y_i) \\ k_2 &= f(x_i + \frac{1}{2}h, y_i + \frac{1}{2} k_1 h) \\ k_3 &= f(x_i + \frac{1}{2}h, y_i + \frac{1}{2} k_2 h ) \\ k_4 &= f(x_i + h, y_i + k_3 h) \end{align} \)$
with \(O(h^5)\) local / \(O(h^4)\) global error.
The Butcher Tableau is: $\( \begin{array}{c|cccc} 0 & & & & \\ \frac{1}{2} & \frac{1}{2} & & & \\ \frac{1}{2} & 0 & \frac{1}{2} & & \\ 1 & 0 & 0 & 1 & \\ \hline & \frac{1}{6} & \frac{1}{3} & \frac{1}{3} & \frac{1}{6} \\ \end{array} \)$
Example - RK4 steps#
An example of the RK4 algorithm is below, showing partial steps inform subsequent steps culminating in very good estimate!
![Runge-Kutta_slopes.svg](data:image/svg+xml;base64,<?xml version="1.0" encoding="UTF-8"?>
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="504pt" height="504pt" viewBox="0 0 504 504" version="1.1">
<defs>
<g>
<symbol overflow="visible" id="glyph0-0">
<path style="stroke:none;" d="M 4.125 -1.257813 L 4.328125 -0.015625 C 3.929688 0.0664063 3.574219 0.105469 3.265625 0.109375 C 2.75 0.105469 2.355469 0.0273438 2.078125 -0.132813 C 1.792969 -0.292969 1.59375 -0.503906 1.484375 -0.769531 C 1.367188 -1.027344 1.3125 -1.582031 1.3125 -2.429688 L 1.3125 -7.203125 L 0.28125 -7.203125 L 0.28125 -8.296875 L 1.3125 -8.296875 L 1.3125 -10.351563 L 2.710938 -11.195313 L 2.710938 -8.296875 L 4.125 -8.296875 L 4.125 -7.203125 L 2.710938 -7.203125 L 2.710938 -2.351563 C 2.707031 -1.945313 2.730469 -1.6875 2.785156 -1.578125 C 2.832031 -1.460938 2.914063 -1.371094 3.027344 -1.304688 C 3.136719 -1.234375 3.296875 -1.199219 3.507813 -1.203125 C 3.660156 -1.199219 3.867188 -1.21875 4.125 -1.257813 Z M 4.125 -1.257813 "/>
</symbol>
<symbol overflow="visible" id="glyph0-1">
<path style="stroke:none;" d="M 1.054688 0 L 1.054688 -11.453125 L 2.460938 -11.453125 L 2.460938 -7.34375 C 3.117188 -8.101563 3.945313 -8.484375 4.945313 -8.484375 C 5.558594 -8.484375 6.089844 -8.363281 6.546875 -8.121094 C 6.996094 -7.878906 7.320313 -7.542969 7.519531 -7.117188 C 7.710938 -6.6875 7.808594 -6.066406 7.8125 -5.257813 L 7.8125 0 L 6.40625 0 L 6.40625 -5.257813 C 6.402344 -5.957031 6.25 -6.46875 5.949219 -6.792969 C 5.640625 -7.109375 5.210938 -7.269531 4.65625 -7.273438 C 4.238281 -7.269531 3.847656 -7.164063 3.480469 -6.949219 C 3.113281 -6.730469 2.851563 -6.4375 2.695313 -6.070313 C 2.539063 -5.699219 2.460938 -5.1875 2.460938 -4.539063 L 2.460938 0 Z M 1.054688 0 "/>
</symbol>
<symbol overflow="visible" id="glyph0-2">
<path style="stroke:none;" d="M 8.054688 -1.351563 L 8.054688 0 L 0.484375 0 C 0.472656 -0.335938 0.527344 -0.660156 0.648438 -0.976563 C 0.839844 -1.488281 1.148438 -1.996094 1.574219 -2.5 C 1.996094 -2.996094 2.609375 -3.574219 3.414063 -4.234375 C 4.65625 -5.253906 5.496094 -6.0625 5.9375 -6.660156 C 6.371094 -7.253906 6.589844 -7.816406 6.59375 -8.351563 C 6.589844 -8.90625 6.390625 -9.378906 5.996094 -9.761719 C 5.59375 -10.144531 5.074219 -10.335938 4.4375 -10.335938 C 3.757813 -10.335938 3.21875 -10.132813 2.8125 -9.726563 C 2.40625 -9.320313 2.199219 -8.757813 2.195313 -8.039063 L 0.75 -8.1875 C 0.847656 -9.261719 1.21875 -10.085938 1.867188 -10.652344 C 2.511719 -11.214844 3.378906 -11.496094 4.46875 -11.5 C 5.566406 -11.496094 6.4375 -11.191406 7.078125 -10.585938 C 7.71875 -9.972656 8.039063 -9.21875 8.039063 -8.320313 C 8.039063 -7.859375 7.945313 -7.40625 7.757813 -6.96875 C 7.570313 -6.523438 7.257813 -6.058594 6.824219 -5.570313 C 6.386719 -5.078125 5.664063 -4.40625 4.65625 -3.554688 C 3.808594 -2.84375 3.265625 -2.363281 3.03125 -2.113281 C 2.789063 -1.859375 2.59375 -1.605469 2.4375 -1.351563 Z M 8.054688 -1.351563 "/>
</symbol>
<symbol overflow="visible" id="glyph0-3">
<path style="stroke:none;" d="M 0.992188 3.195313 L 0.835938 1.875 C 1.140625 1.957031 1.410156 1.996094 1.640625 2 C 1.953125 1.996094 2.203125 1.945313 2.390625 1.84375 C 2.578125 1.738281 2.730469 1.59375 2.851563 1.40625 C 2.9375 1.265625 3.082031 0.914063 3.28125 0.359375 C 3.304688 0.277344 3.34375 0.164063 3.40625 0.015625 L 0.257813 -8.296875 L 1.773438 -8.296875 L 3.5 -3.492188 C 3.71875 -2.882813 3.921875 -2.242188 4.101563 -1.570313 C 4.261719 -2.214844 4.453125 -2.84375 4.679688 -3.460938 L 6.453125 -8.296875 L 7.859375 -8.296875 L 4.703125 0.140625 C 4.363281 1.050781 4.101563 1.675781 3.914063 2.023438 C 3.664063 2.484375 3.375 2.824219 3.054688 3.042969 C 2.726563 3.253906 2.34375 3.363281 1.898438 3.367188 C 1.625 3.363281 1.320313 3.304688 0.992188 3.195313 Z M 0.992188 3.195313 "/>
</symbol>
<symbol overflow="visible" id="glyph0-4">
<path style="stroke:none;" d="M 1.0625 0 L 1.0625 -11.453125 L 2.46875 -11.453125 L 2.46875 -4.921875 L 5.796875 -8.296875 L 7.617188 -8.296875 L 4.445313 -5.21875 L 7.9375 0 L 6.203125 0 L 3.460938 -4.242188 L 2.46875 -3.289063 L 2.46875 0 Z M 1.0625 0 "/>
</symbol>
<symbol overflow="visible" id="glyph0-5">
<path style="stroke:none;" d="M 3.742188 3.367188 C 2.964844 2.386719 2.308594 1.242188 1.773438 -0.0703125 C 1.234375 -1.382813 0.964844 -2.742188 0.96875 -4.148438 C 0.964844 -5.386719 1.167969 -6.574219 1.570313 -7.710938 C 2.039063 -9.027344 2.761719 -10.339844 3.742188 -11.648438 L 4.75 -11.648438 C 4.117188 -10.5625 3.699219 -9.789063 3.5 -9.328125 C 3.179688 -8.609375 2.929688 -7.859375 2.75 -7.078125 C 2.523438 -6.101563 2.414063 -5.121094 2.414063 -4.140625 C 2.414063 -1.632813 3.191406 0.867188 4.75 3.367188 Z M 3.742188 3.367188 "/>
</symbol>
<symbol overflow="visible" id="glyph0-6">
<path style="stroke:none;" d="M 1.976563 3.367188 L 0.96875 3.367188 C 2.523438 0.867188 3.304688 -1.632813 3.304688 -4.140625 C 3.304688 -5.117188 3.191406 -6.089844 2.96875 -7.054688 C 2.789063 -7.835938 2.542969 -8.585938 2.226563 -9.304688 C 2.023438 -9.773438 1.601563 -10.554688 0.96875 -11.648438 L 1.976563 -11.648438 C 2.953125 -10.339844 3.679688 -9.027344 4.148438 -7.710938 C 4.546875 -6.574219 4.746094 -5.386719 4.75 -4.148438 C 4.746094 -2.742188 4.476563 -1.382813 3.941406 -0.0703125 C 3.398438 1.242188 2.742188 2.386719 1.976563 3.367188 Z M 1.976563 3.367188 "/>
</symbol>
<symbol overflow="visible" id="glyph0-7">
<path style="stroke:none;" d="M 1.421875 0 L 1.421875 -1.601563 L 3.023438 -1.601563 L 3.023438 0 C 3.023438 0.585938 2.917969 1.058594 2.710938 1.425781 C 2.5 1.785156 2.167969 2.066406 1.71875 2.265625 L 1.328125 1.664063 C 1.621094 1.53125 1.839844 1.339844 1.984375 1.089844 C 2.121094 0.835938 2.199219 0.472656 2.21875 0 Z M 1.421875 0 "/>
</symbol>
<symbol overflow="visible" id="glyph0-8">
<path style="stroke:none;" d=""/>
</symbol>
<symbol overflow="visible" id="glyph1-0">
<path style="stroke:none;" d="M 0.464844 -3.953125 C 0.460938 -4.898438 0.558594 -5.660156 0.757813 -6.242188 C 0.949219 -6.816406 1.238281 -7.261719 1.625 -7.578125 C 2.007813 -7.890625 2.492188 -8.050781 3.078125 -8.050781 C 3.503906 -8.050781 3.882813 -7.960938 4.210938 -7.789063 C 4.535156 -7.613281 4.804688 -7.363281 5.015625 -7.039063 C 5.226563 -6.710938 5.390625 -6.316406 5.511719 -5.847656 C 5.628906 -5.378906 5.6875 -4.746094 5.691406 -3.953125 C 5.6875 -3.011719 5.59375 -2.253906 5.402344 -1.675781 C 5.207031 -1.097656 4.917969 -0.648438 4.535156 -0.335938 C 4.152344 -0.0195313 3.664063 0.132813 3.078125 0.136719 C 2.300781 0.132813 1.695313 -0.140625 1.257813 -0.695313 C 0.726563 -1.359375 0.460938 -2.445313 0.464844 -3.953125 Z M 1.476563 -3.953125 C 1.476563 -2.636719 1.628906 -1.761719 1.9375 -1.324219 C 2.246094 -0.886719 2.625 -0.667969 3.078125 -0.671875 C 3.527344 -0.667969 3.910156 -0.886719 4.21875 -1.328125 C 4.523438 -1.761719 4.675781 -2.636719 4.679688 -3.953125 C 4.675781 -5.269531 4.523438 -6.144531 4.21875 -6.582031 C 3.910156 -7.011719 3.523438 -7.230469 3.066406 -7.234375 C 2.613281 -7.230469 2.253906 -7.039063 1.984375 -6.660156 C 1.644531 -6.171875 1.476563 -5.269531 1.476563 -3.953125 Z M 1.476563 -3.953125 "/>
</symbol>
<symbol overflow="visible" id="glyph1-1">
<path style="stroke:none;" d="M 4.171875 0 L 3.1875 0 L 3.1875 -6.273438 C 2.949219 -6.042969 2.636719 -5.816406 2.253906 -5.59375 C 1.867188 -5.363281 1.523438 -5.195313 1.21875 -5.085938 L 1.21875 -6.039063 C 1.769531 -6.292969 2.25 -6.605469 2.664063 -6.976563 C 3.074219 -7.34375 3.367188 -7.703125 3.539063 -8.050781 L 4.171875 -8.050781 Z M 4.171875 0 "/>
</symbol>
<symbol overflow="visible" id="glyph1-2">
<path style="stroke:none;" d="M 5.636719 -0.945313 L 5.636719 0 L 0.339844 0 C 0.332031 -0.234375 0.367188 -0.460938 0.453125 -0.683594 C 0.585938 -1.042969 0.804688 -1.398438 1.101563 -1.75 C 1.398438 -2.097656 1.828125 -2.503906 2.390625 -2.964844 C 3.257813 -3.675781 3.84375 -4.238281 4.15625 -4.660156 C 4.460938 -5.074219 4.617188 -5.472656 4.617188 -5.847656 C 4.617188 -6.234375 4.476563 -6.5625 4.195313 -6.832031 C 3.914063 -7.097656 3.550781 -7.230469 3.105469 -7.234375 C 2.628906 -7.230469 2.25 -7.089844 1.96875 -6.808594 C 1.683594 -6.523438 1.539063 -6.128906 1.535156 -5.628906 L 0.523438 -5.730469 C 0.589844 -6.484375 0.851563 -7.058594 1.304688 -7.457031 C 1.753906 -7.851563 2.363281 -8.050781 3.128906 -8.050781 C 3.894531 -8.050781 4.5 -7.835938 4.953125 -7.410156 C 5.398438 -6.980469 5.625 -6.453125 5.628906 -5.824219 C 5.625 -5.503906 5.558594 -5.1875 5.429688 -4.878906 C 5.296875 -4.566406 5.078125 -4.238281 4.777344 -3.898438 C 4.46875 -3.554688 3.964844 -3.085938 3.257813 -2.488281 C 2.667969 -1.992188 2.289063 -1.65625 2.121094 -1.480469 C 1.953125 -1.300781 1.8125 -1.121094 1.707031 -0.945313 Z M 5.636719 -0.945313 "/>
</symbol>
<symbol overflow="visible" id="glyph1-3">
<path style="stroke:none;" d="M 0.46875 -2.117188 L 1.453125 -2.246094 C 1.566406 -1.683594 1.757813 -1.28125 2.03125 -1.039063 C 2.300781 -0.789063 2.632813 -0.667969 3.023438 -0.671875 C 3.484375 -0.667969 3.875 -0.828125 4.195313 -1.152344 C 4.515625 -1.472656 4.675781 -1.871094 4.675781 -2.347656 C 4.675781 -2.796875 4.527344 -3.167969 4.230469 -3.464844 C 3.933594 -3.753906 3.558594 -3.902344 3.105469 -3.90625 C 2.917969 -3.902344 2.683594 -3.867188 2.410156 -3.796875 L 2.519531 -4.660156 C 2.585938 -4.648438 2.636719 -4.644531 2.679688 -4.648438 C 3.09375 -4.644531 3.472656 -4.753906 3.8125 -4.976563 C 4.144531 -5.191406 4.3125 -5.53125 4.316406 -5.988281 C 4.3125 -6.347656 4.191406 -6.644531 3.949219 -6.886719 C 3.703125 -7.121094 3.386719 -7.242188 3.003906 -7.242188 C 2.617188 -7.242188 2.296875 -7.121094 2.046875 -6.878906 C 1.789063 -6.636719 1.625 -6.273438 1.554688 -5.796875 L 0.570313 -5.972656 C 0.6875 -6.628906 0.960938 -7.140625 1.386719 -7.503906 C 1.8125 -7.867188 2.34375 -8.050781 2.980469 -8.050781 C 3.414063 -8.050781 3.816406 -7.957031 4.1875 -7.769531 C 4.554688 -7.582031 4.835938 -7.324219 5.035156 -7 C 5.226563 -6.671875 5.324219 -6.328125 5.328125 -5.964844 C 5.324219 -5.617188 5.230469 -5.300781 5.046875 -5.019531 C 4.859375 -4.734375 4.585938 -4.507813 4.222656 -4.34375 C 4.695313 -4.230469 5.0625 -4.003906 5.324219 -3.660156 C 5.585938 -3.3125 5.71875 -2.882813 5.71875 -2.367188 C 5.71875 -1.664063 5.460938 -1.070313 4.953125 -0.585938 C 4.4375 -0.101563 3.792969 0.140625 3.019531 0.140625 C 2.3125 0.140625 1.730469 -0.0664063 1.265625 -0.488281 C 0.800781 -0.902344 0.535156 -1.445313 0.46875 -2.117188 Z M 0.46875 -2.117188 "/>
</symbol>
<symbol overflow="visible" id="glyph1-4">
<path style="stroke:none;" d="M 3.621094 0 L 3.621094 -1.917969 L 0.140625 -1.917969 L 0.140625 -2.820313 L 3.800781 -8.015625 L 4.605469 -8.015625 L 4.605469 -2.820313 L 5.6875 -2.820313 L 5.6875 -1.917969 L 4.605469 -1.917969 L 4.605469 0 Z M 3.621094 -2.820313 L 3.621094 -6.4375 L 1.109375 -2.820313 Z M 3.621094 -2.820313 "/>
</symbol>
<symbol overflow="visible" id="glyph2-0">
<path style="stroke:none;" d="M 4.015625 -8.203125 L 4.78125 -8.203125 L 4.78125 -4.46875 L 8.492188 -4.46875 L 8.492188 -3.710938 L 4.78125 -3.710938 L 4.78125 0 L 4.015625 0 L 4.015625 -3.710938 L 0.289063 -3.710938 L 0.289063 -4.46875 L 4.015625 -4.46875 Z M 4.015625 -8.203125 "/>
</symbol>
<symbol overflow="visible" id="glyph3-0">
<path style="stroke:none;" d="M 6.210938 3.898438 L 6.210938 4.316406 C 4.511719 3.46875 3.1875 2.226563 2.242188 0.589844 C 1.292969 -1.046875 0.820313 -2.835938 0.820313 -4.785156 C 0.820313 -6.804688 1.316406 -8.648438 2.3125 -10.316406 C 3.308594 -11.976563 4.609375 -13.167969 6.210938 -13.886719 L 6.210938 -13.476563 C 5.40625 -13.03125 4.746094 -12.425781 4.238281 -11.660156 C 3.722656 -10.890625 3.339844 -9.914063 3.085938 -8.734375 C 2.828125 -7.550781 2.699219 -6.320313 2.703125 -5.039063 C 2.699219 -3.585938 2.816406 -2.273438 3.054688 -1.101563 C 3.285156 0.0703125 3.644531 1.039063 4.128906 1.8125 C 4.609375 2.578125 5.304688 3.273438 6.210938 3.898438 Z M 6.210938 3.898438 "/>
</symbol>
<symbol overflow="visible" id="glyph3-1">
<path style="stroke:none;" d="M 0.4375 -13.476563 L 0.4375 -13.886719 C 2.136719 -13.042969 3.460938 -11.800781 4.40625 -10.167969 C 5.351563 -8.527344 5.824219 -6.738281 5.828125 -4.796875 C 5.824219 -2.769531 5.328125 -0.925781 4.335938 0.742188 C 3.339844 2.40625 2.039063 3.597656 0.4375 4.316406 L 0.4375 3.898438 C 1.242188 3.453125 1.902344 2.847656 2.421875 2.082031 C 2.933594 1.308594 3.316406 0.339844 3.570313 -0.839844 C 3.816406 -2.015625 3.941406 -3.25 3.945313 -4.539063 C 3.941406 -5.984375 3.828125 -7.296875 3.597656 -8.472656 C 3.363281 -9.644531 3.003906 -10.617188 2.523438 -11.390625 C 2.035156 -12.160156 1.339844 -12.855469 0.4375 -13.476563 Z M 0.4375 -13.476563 "/>
</symbol>
</g>
<clipPath id="clip1">
  <path d="M 21.121094 21.121094 L 483.882813 21.121094 L 483.882813 483.882813 L 21.121094 483.882813 Z M 21.121094 21.121094 "/>
</clipPath>
</defs>
<g id="surface61">
<rect x="0" y="0" width="504" height="504" style="fill:rgb(100%,100%,100%);fill-opacity:1;stroke:none;"/>
<g clip-path="url(#clip1)" clip-rule="nonzero">
<path style="fill:none;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,100%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 113.042969 337.984375 L 116.605469 336.496094 L 120.171875 334.980469 L 123.734375 333.445313 L 127.296875 331.886719 L 130.859375 330.300781 L 134.421875 328.695313 L 137.984375 327.058594 L 141.546875 325.402344 L 145.109375 323.714844 L 148.675781 322.003906 L 152.238281 320.265625 L 155.800781 318.496094 L 159.363281 316.699219 L 162.925781 314.871094 L 166.488281 313.011719 L 170.050781 311.121094 L 173.613281 309.199219 L 177.179688 307.242188 L 180.742188 305.25 L 184.304688 303.222656 L 187.867188 301.15625 L 191.429688 299.050781 L 194.992188 296.90625 L 198.554688 294.722656 L 202.117188 292.496094 L 205.679688 290.226563 L 209.246094 287.910156 L 212.808594 285.546875 L 216.371094 283.136719 L 219.933594 280.675781 L 223.496094 278.164063 L 227.058594 275.601563 L 230.621094 272.980469 L 234.183594 270.304688 L 237.75 267.570313 L 241.3125 264.773438 L 244.875 261.914063 L 248.4375 258.992188 L 252 256 L 255.5625 252.941406 L 259.125 249.8125 L 262.6875 246.605469 L 266.25 243.328125 L 269.816406 239.96875 L 273.378906 236.53125 L 276.941406 233.007813 L 280.503906 229.398438 L 284.066406 225.703125 L 287.628906 221.914063 L 291.191406 218.03125 L 294.753906 214.050781 L 298.320313 209.96875 L 301.882813 205.785156 L 305.445313 201.492188 L 309.007813 197.089844 L 312.570313 192.578125 L 316.132813 187.945313 L 319.695313 183.195313 L 323.257813 178.324219 L 326.820313 173.320313 L 330.386719 168.1875 L 333.949219 162.921875 L 337.511719 157.515625 L 341.074219 151.96875 L 344.636719 146.273438 L 348.199219 140.425781 L 351.761719 134.425781 L 355.324219 128.265625 L 358.890625 121.941406 L 362.453125 115.449219 L 366.015625 108.785156 L 369.578125 101.941406 L 373.140625 94.914063 L 376.703125 87.699219 L 380.265625 80.292969 L 383.828125 72.691406 L 387.394531 64.882813 L 390.957031 56.867188 L 394.519531 48.636719 L 398.082031 40.1875 L 401.644531 31.511719 L 405.207031 22.609375 L 408.769531 13.464844 L 412.332031 4.078125 L 413.84375 0 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 85.726563 436.625 L 465.777344 436.625 "/>
<path style="fill-rule:nonzero;fill:rgb(0%,0%,0%);fill-opacity:1;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 458.480469 432.617188 L 461.609375 434.320313 L 465.175781 435.585938 L 469.046875 436.363281 L 473.074219 436.625 L 469.046875 436.890625 L 465.175781 437.667969 L 461.609375 438.933594 L 458.480469 440.636719 L 458.816406 440.535156 L 459.136719 440.238281 L 459.425781 439.761719 L 459.664063 439.125 L 459.84375 438.367188 L 459.953125 437.519531 L 459.992188 436.625 L 459.953125 435.734375 L 459.84375 434.886719 L 459.664063 434.125 L 459.425781 433.492188 L 459.136719 433.011719 L 458.816406 432.714844 Z M 458.480469 432.617188 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 446.34375 L 109.480469 38.222656 "/>
<path style="fill-rule:nonzero;fill:rgb(0%,0%,0%);fill-opacity:1;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 105.472656 45.519531 L 107.175781 42.390625 L 108.4375 38.824219 L 109.21875 34.953125 L 109.480469 30.925781 L 109.746094 34.953125 L 110.523438 38.824219 L 111.789063 42.390625 L 113.492188 45.519531 L 113.390625 45.183594 L 113.09375 44.863281 L 112.617188 44.574219 L 111.980469 44.335938 L 111.222656 44.15625 L 110.375 44.046875 L 109.480469 44.007813 L 108.589844 44.046875 L 107.742188 44.15625 L 106.980469 44.335938 L 106.347656 44.574219 L 105.867188 44.863281 L 105.570313 45.183594 Z M 105.472656 45.519531 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 109.480469 339.453125 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 228.246094 290.867188 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 228.246094 260.503906 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 347.011719 151.183594 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 347.011719 144.097656 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(49.803922%,49.803922%,49.803922%);stroke-opacity:1;stroke-dasharray:3,3;stroke-miterlimit:10;" d="M 228.246094 436.625 L 228.246094 260.503906 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(49.803922%,49.803922%,49.803922%);stroke-opacity:1;stroke-dasharray:3,3;stroke-miterlimit:10;" d="M 347.011719 436.625 L 347.011719 151.183594 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(49.803922%,49.803922%,49.803922%);stroke-opacity:1;stroke-dasharray:3,3;stroke-miterlimit:10;" d="M 109.480469 290.867188 L 228.246094 290.867188 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(49.803922%,49.803922%,49.803922%);stroke-opacity:1;stroke-dasharray:3,3;stroke-miterlimit:10;" d="M 109.480469 260.503906 L 228.246094 260.503906 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(49.803922%,49.803922%,49.803922%);stroke-opacity:1;stroke-dasharray:3,3;stroke-miterlimit:10;" d="M 109.480469 151.183594 L 347.011719 151.183594 "/>
<path style="fill:none;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 109.480469 339.453125 L 168.863281 315.160156 "/>
<path style="fill:none;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 228.246094 290.867188 L 287.628906 251.394531 "/>
<path style="fill:none;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 228.246094 260.503906 L 287.628906 213.433594 "/>
<path style="fill:none;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 347.011719 151.183594 L 406.394531 55.53125 "/>
<path style="fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 160.59375 314.210938 L 164.132813 314.605469 L 167.914063 314.425781 L 171.792969 313.679688 L 175.617188 312.398438 L 171.992188 314.167969 L 168.703125 316.355469 L 165.878906 318.875 L 163.628906 321.636719 L 163.902344 321.414063 L 164.085938 321.019531 L 164.171875 320.46875 L 164.152344 319.789063 L 164.03125 319.019531 L 163.8125 318.191406 L 163.511719 317.351563 L 163.136719 316.539063 L 162.710938 315.796875 L 162.257813 315.164063 L 161.796875 314.664063 L 161.351563 314.332031 L 160.941406 314.179688 Z M 160.59375 314.210938 "/>
<path style="fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 279.332031 252.09375 L 282.878906 251.777344 L 286.550781 250.859375 L 290.207031 249.363281 L 293.707031 247.355469 L 290.5 249.800781 L 287.707031 252.59375 L 285.433594 255.621094 L 283.773438 258.769531 L 284 258.5 L 284.101563 258.078125 L 284.074219 257.519531 L 283.921875 256.859375 L 283.652344 256.128906 L 283.273438 255.359375 L 282.8125 254.59375 L 282.285156 253.875 L 281.722656 253.230469 L 281.152344 252.695313 L 280.601563 252.300781 L 280.097656 252.058594 L 279.667969 251.988281 Z M 279.332031 252.09375 "/>
<path style="fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 279.421875 214.824219 L 282.929688 214.214844 L 286.511719 212.992188 L 290.027344 211.199219 L 293.347656 208.902344 L 290.355469 211.609375 L 287.804688 214.625 L 285.796875 217.832031 L 284.402344 221.109375 L 284.605469 220.820313 L 284.667969 220.390625 L 284.597656 219.839844 L 284.390625 219.191406 L 284.058594 218.484375 L 283.621094 217.75 L 283.097656 217.027344 L 282.511719 216.351563 L 281.898438 215.757813 L 281.285156 215.273438 L 280.703125 214.925781 L 280.179688 214.726563 L 279.746094 214.695313 Z M 279.421875 214.824219 "/>
<path style="fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;stroke-width:1.5;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(80.392157%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 399.140625 59.613281 L 402.234375 57.855469 L 405.191406 55.492188 L 407.894531 52.613281 L 410.242188 49.332031 L 408.34375 52.890625 L 406.964844 56.589844 L 406.15625 60.289063 L 405.953125 63.84375 L 406.046875 63.507813 L 405.960938 63.078125 L 405.707031 62.582031 L 405.292969 62.042969 L 404.742188 61.492188 L 404.082031 60.949219 L 403.34375 60.445313 L 402.566406 60.007813 L 401.785156 59.65625 L 401.046875 59.40625 L 400.378906 59.277344 L 399.824219 59.265625 L 399.402344 59.382813 Z M 399.140625 59.613281 "/>
<path style=" stroke:none;fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;" d="M 113.082031 339.453125 C 113.082031 344.253906 105.882813 344.253906 105.882813 339.453125 C 105.882813 334.65625 113.082031 334.65625 113.082031 339.453125 "/>
<path style=" stroke:none;fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;" d="M 231.847656 290.867188 C 231.847656 295.667969 224.648438 295.667969 224.648438 290.867188 C 224.648438 286.070313 231.847656 286.070313 231.847656 290.867188 "/>
<path style=" stroke:none;fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;" d="M 231.847656 260.503906 C 231.847656 265.300781 224.648438 265.300781 224.648438 260.503906 C 224.648438 255.703125 231.847656 255.703125 231.847656 260.503906 "/>
<path style=" stroke:none;fill-rule:nonzero;fill:rgb(80.392157%,0%,0%);fill-opacity:1;" d="M 350.613281 151.183594 C 350.613281 155.984375 343.414063 155.984375 343.414063 151.183594 C 343.414063 146.382813 350.613281 146.382813 350.613281 151.183594 "/>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-0" x="104.144531" y="464.765625"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="108.589844" y="467.667969"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-0" x="202.511719" y="466.941406"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="206.957031" y="469.847656"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph2-0" x="216.148438" y="466.941406"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-1" x="227.890625" y="466.941406"/>
</g>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 238.863281 469.296875 L 243.011719 453.132813 "/>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-2" x="245.085938" y="466.941406"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-0" x="329.875" y="464.980469"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="334.320313" y="467.882813"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph2-0" x="343.507813" y="464.980469"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-1" x="355.253906" y="464.980469"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="85.652344" y="339.976563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="93.652344" y="342.878906"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="30.625" y="293.226563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="38.625" y="296.128906"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph2-0" x="47.816406" y="293.226563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-1" x="59.558594" y="293.226563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="68.457031" y="293.226563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-1" x="76.457031" y="296.128906"/>
</g>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 84.761719 295.582031 L 88.910156 279.417969 "/>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-2" x="90.984375" y="293.226563"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="30.625" y="262.859375"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="38.625" y="265.765625"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph2-0" x="47.816406" y="262.859375"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-1" x="59.558594" y="262.859375"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="68.457031" y="262.859375"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-2" x="76.457031" y="265.765625"/>
</g>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,0%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 84.761719 265.214844 L 88.910156 249.050781 "/>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-2" x="90.984375" y="262.859375"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="47.820313" y="152.757813"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-0" x="55.820313" y="155.660156"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph2-0" x="65.011719" y="152.757813"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-1" x="76.753906" y="152.757813"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="85.652344" y="152.757813"/>
</g>
<g style="fill:rgb(0%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-3" x="93.652344" y="155.660156"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="126.121094" y="348.398438"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-1" x="134.121094" y="351.300781"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="252.011719" y="290.09375"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-2" x="260.011719" y="293"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="242.511719" y="231.335938"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-3" x="250.511719" y="234.242188"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-4" x="375.527344" y="121.257813"/>
</g>
<g style="fill:rgb(80.392157%,0%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-4" x="383.527344" y="124.164063"/>
</g>
<g style="fill:rgb(0%,0%,100%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="361.816406" y="52.453125"/>
  <use xlink:href="#glyph0-5" x="369.816406" y="52.453125"/>
  <use xlink:href="#glyph0-0" x="375.144531" y="52.453125"/>
  <use xlink:href="#glyph0-6" x="379.589844" y="52.453125"/>
</g>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,54.509804%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 343.414063 147.699219 L 350.613281 140.5 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,54.509804%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 343.414063 140.5 L 350.613281 147.699219 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,54.509804%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 341.921875 144.097656 L 352.101563 144.097656 "/>
<path style="fill:none;stroke-width:0.75;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(0%,54.509804%,0%);stroke-opacity:1;stroke-miterlimit:10;" d="M 347.011719 149.191406 L 347.011719 139.007813 "/>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph3-0" x="290.296875" y="136.132813"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-0" x="296.957031" y="136.132813"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-1" x="301.402344" y="139.035156"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-7" x="307.632813" y="136.132813"/>
  <use xlink:href="#glyph0-8" x="312.078125" y="136.132813"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph0-3" x="316.523438" y="136.132813"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph1-1" x="324.523438" y="139.035156"/>
</g>
<g style="fill:rgb(0%,54.509804%,0%);fill-opacity:1;">
  <use xlink:href="#glyph3-1" x="330.753906" y="136.132813"/>
</g>
</g>
</g>
</svg>
)
Adaptive time stepping methods#
It is not uncommon for systems to have a combination of fast transients with otherwise slow behaviour. In general, we won’t know when the transient occurs, or even it’s rate of change!
E.g.: The system \(y(t) = \tanh(\frac{t-5}{0.1})-t\) has a slow decrease with a sharp bump. If the time step is small, the bump will be resolved, but it will take a long time for the quasi-linear parts. If the step is too large, the bump may be missed entirely!
import numpy as np
import matplotlib.pyplot as plt
# Create a range of t values
t = np.linspace(0, 10, 100)
# Plot the function
plt.plot(t, np.tanh((t-5)/.1) - t)
plt.xlabel('t')
plt.ylabel('y(t)')
plt.grid(True)
plt.show()
Adaptive time steppers automatically adjust their step size based on an estimate of the error.
If your timestep has an error \(E_{current}\) with stepsize \(h_{current}\), and you are aiming for an error tolerance \(E_{goal}\) then you can update your step size with a formula like:
where \(\alpha\) is some user value (0.2 is a good choice).
There are remaining quesitons though:
Would you reject the time step if \(E_{current} \gt E_{tolerance}\) and repeat with a new timestep?
Would you have a meta analysis of the solver method to ‘catch’ non-convergent cases?
There are a few methods to capture the error. Naturally, you don’t want the error calculation to be overly burdomsome.
Change step-size
Change integration order
Directly compare \(y_i\) to \(y_{i+1}\) or other properties of the solution.
Step-halving methods#
Step-halving methods compare the results of a full step to two half steps. For RK4, the error is calculated as:
for this RK4 scheme, one could correct higher accuracy estimate with $\(y_{double \ step}^* = y_{double \ step} + \frac{E}{15}\)$
Example: Integrate $\(y^\prime = 4 e^{0.8x}-0.5y\)\( with \)y(0) = 2\( from \)x = 0\ to\ 2$. The analytic answer is 14.84392.
# prompt: solve above with RK4
import numpy as np
import matplotlib.pyplot as plt
def f(x, y):
"""The differential equation."""
return 4 * np.exp(0.8 * x) - 0.5 * y
def rk4_step(f, x, y, h):
"""Takes a single RK4 step."""
k1 = h * f(x, y)
k2 = h * f(x + h/2, y + k1/2)
k3 = h * f(x + h/2, y + k2/2)
k4 = h * f(x + h, y + k3)
return y + (k1 + 2*k2 + 2*k3 + k4) / 6
# Initial conditions
x0 = 0
y0 = 2
x_end = 2
h_initial = 2
y_onestep = rk4_step(f, x0, y0, 2)
print('One stpe with h=2 ', y_onestep)
y_t = rk4_step(f, x0, y0, 1)
y_twostep = rk4_step(f, x0+1, y_t, 1)
print('Two steps with h=1 ', y_twostep)
print('Approximate error is ', (y_twostep - y_onestep)/15, ' vs true error, ', 14.84392 - y_twostep)
One stpe with h=2 15.105846327501714
Two steps with h=1 14.8624835881192
Approximate error is -0.016224182625500915 vs true error, -0.018563588119199892
Note that to calculate this value we had to evaluate \(f(x,y)\) 8 times.
Runge-Kutta Fehlberg#
Another approach is to compare different integration orders over the same step. This is made efficient by reusing function calls between the two approximations.
where the doubling of the last line means:
The error is then simply \(y_{i+1}^{(5)} - y_{i+1}^{(4)}\)
This is the method of choice for packaged tools. E.g.:
# prompt: solve 4 * np.exp(0.8 * x) - 0.5 * y with scipy , step size 2 and output the error
import numpy as np
from scipy.integrate import solve_ivp
def f(x, y):
"""The differential equation."""
return 4 * np.exp(0.8 * x) - 0.5 * y
# Initial conditions
x0 = 0
y0 = 2
x_end = 2
# Solve the differential equation using solve_ivp
sol = solve_ivp(f, (x0, x_end), [y0], method='RK45')
print(sol)
# Extract the solution
y_numerical = sol.y[0][-1]
# Analytical solution (you might need to calculate this beforehand)
y_analytical = 14.84392
# Calculate the error
error = abs(y_numerical - y_analytical)
print(f"Numerical solution: {y_numerical}")
print(f"Error: {error}")
message: The solver successfully reached the end of the integration interval.
success: True
status: 0
t: [ 0.000e+00 9.222e-02 1.014e+00 2.000e+00]
y: [[ 2.000e+00 2.284e+00 6.279e+00 1.484e+01]]
sol: None
t_events: None
y_events: None
nfev: 20
njev: 0
nlu: 0
Numerical solution: 14.844062517715555
Error: 0.00014251771555429116
Direct comparison#
A less elegant, but sometimes more pragmatic method is use other metrics to control the step size.
Following a time step, the rate of change can be calculated,
$\(\frac{dy}{dt} = \frac{y_{i+1}-y_i}{h}\)\(
from which a control on \)|\frac{dy}{dt}|\( can be placed. E.g.: one could require that the Euclidian or \)\infty$ norm be below a tolerance. This method controls the change in solution, rather than its error.
An even more brute-force method may be to observe the efficiency of the solver. We konw that nonlinear root finding with Newton’s method converges quadratically near the root. By monitoring the Newton iterations at each timestep, one can assess if the solver is converging quadratically, or is taking too many iterations which may mean the new time step is too far from the previous one.