<html>
<head></head>
<body bgcolor="white">

<title>Numerical investigations</title>

<center><h1>Experiments with Schemes for Exponential Decay</h1></center>

<center><b>Hans Petter Langtangen</b></center>
<center><b>Simula Research Laboratory</b></center>
<center><b>University of Oslo</b></center>
<center><h4>August 20, 2012</h4></center>

<b>Summary.</b> This report investigates the accuracy of three
finite difference schemes for the ordinary differential equation
u'=-au with the aid of numerical experiments.  Numerical
artifacts are in particular demonstrated.

<h2>Mathematical problem</h2>

We address the initial-value problem u'(t)=-au(t),
for t in (0,T], and with initial condition u(0)=I.
Here, I, a, and T are prescribed parameters,
and u(t) is the unknown function to be estimated.
This mathematical model is relevant for physical
phenomena featuring exponential decay in time,
e.g., vertical pressure variation in the atmosphere,
cooling of an object, and radioactive decay.

<h2>Numerical solution method</h2>

We use a mesh in time with equally spaced points.
The theta-rule <a href="#Iserles_2009">[1]</a>
is used to discretize the differential
equation. This scheme corresponds to

<ul>
  <li> The Forward Euler scheme when theta=0
  <li> The Backward Euler scheme when theta=1
  <li> The Crank-Nicolson scheme when theta=1/2
</ul>

<h2>Implementation</h2>

The numerical method is implemented in a Python function
<a href="#Langtangen_2012">[2]</a>:

<pre>
def solver(I, a, T, dt, theta):
    """Solve u'=-a*u, u(0)=I, for t in (0,T] with steps of dt."""
    Nt = int(round(T/float(dt)))  # no of intervals
    u = zeros(Nt+1)
    t = linspace(0, T, Nt+1)

    u[0] = I
    for n in range(0, Nt):
        u[n+1] = (1 - (1-theta)*a*dt)/(1 + theta*dt*a)*u[n]
    return u, t
</pre>

<h2>Numerical experiments</h2>

We define a set of numerical experiments where I, a, and
T are fixed, while the time step and theta are varied.
In particular, I=1, a=2, dt=1.25, 0.75, 0.5, 0.1.


<h3>The Backward Euler method</h3>
<img src="BE.png" width="800">


<h3>The Crank-Nicolson method</h3>
<img src="CN.png" width="800">


<h3>The Forward Euler method</h3>
<img src="FE.png" width="800">



<h3>Error versus time discretization</h3>

How E varies with the time step, for the three discretization
methods, is shown below.

<p><b>Observe:</b>
The data points for the three largest time steps in the
Forward Euler method are not relevant as the solution behaves
non-physically.

<p>
<img="error.png", width="400">

<h3>Summary</h3>

<ol>
  <li> For the Backward Euler scheme, we have
       first-order convergence.
  <li> For the Crank-Nicolson scheme, we have second-order convergence.
  <li> The Backward Euler scheme is always stable and gives
       qualitatively corrects results.
  <li> The Crank-Nicolson scheme never blows up, but may give
       oscillating solutions if the time step is not sufficiently small.
  <li> The Forward Euler scheme suffers from fast-growing solution
       if time time step is not small enough, but even below this
       limit one can have oscillating solutions (unless
       the time step is sufficiently small).
</ol>

<h3>Bibliography</h3>

<ol>
 <li> <a name="Iserles_2009"></a> <b>A. Iserles</b>.
    A First Course in the Numerical Analysis of Differential Equations,
    Cambridge University Press, 2009.
 <li> <a name="Langtangen_2012"></a> <b>H. P. Langtangen</b>.
    A Primer on Scientific Programming With Python,
    Springer, 2012.
</ol>

</body>
</html>