$$ \newcommand{\Oof}[1]{\mathcal{O}(#1)} \newcommand{\F}{\boldsymbol{F}} \newcommand{\J}{\boldsymbol{J}} \newcommand{\x}{\boldsymbol{x}} \renewcommand{\c}{\boldsymbol{c}} $$

Finite difference methods

We shall now construct a numerical method for the diffusion equation. We know how to solve ordinary differential equations, so in a way we are able to deal with the time derivative. Very often in mathematics, a new problem can be solved by reducing it to a series of problems we know how to solve. In the present case, it means that we must do something with the spatial derivative \( \partial^2 /\partial x^2 \) in order to reduce the partial differential equation to ordinary differential equations. One important technique for achieving this, is based on finite difference discretization of spatial derivatives.

Reduction of a PDE to a system of ODEs

Introduce a spatial mesh in \( \Omega \) with mesh points $$ x_0=0 < x_1 < x_2 < \cdots < x_N=L \thinspace .$$ The space between two mesh points \( x_i \) and \( x_{i+1} \), i.e. the interval \( [x_i,x_{i+1}] \), is call a cell. We shall here, for simplicity, assume that each cell has the same length \( \Delta x = x_{i+1}-x_i \), \( i=0,\ldots, N-1 \).

The partial differential equation is valid at all spatial points \( x\in\Omega \), but we may relax this condition and demand that it is fulfilled at the internal mesh points only, \( x_1,\ldots,x_{N-1} \): $$ \begin{equation} \frac{\partial u(x_i,t)}{\partial t} = \beta \frac{\partial^{2}u(x_i,t)}{\partial x^2} + g(x_i,t),\quad i=1,\ldots,N-1 \thinspace . \tag{118} \end{equation} $$ Now, at any point \( x_i \) we can approximate the second-order derivative by a finite difference: $$ \begin{equation} \frac{\partial^{2}u(x_i,t)}{\partial x^2} \approx \frac{u(x_{i+1},t) - 2u(x_i,t) + u(x_{i-1},t)}{\Delta x^2}\thinspace . \tag{119} \end{equation} $$ It is common to introduce a short notation \( u_i(t) \) for \( u(x_i,t) \), i.e., \( u \) approximated at some mesh point \( x_i \) in space. With this new notation we can, after inserting (119) in (118), write an approximation to the partial differential equation at mesh point \( (x_i,t \)) as $$ \begin{equation} \frac{d u_i(t)}{d t} = \beta \frac{u_{i+1}(t) - 2u_i(t) + u_{i-1}(t)}{\Delta x^2} + g_i(t),\quad i=1,\ldots,N-1 \thinspace . \tag{120} \end{equation} $$ Note that we have adopted the notation \( g_i(t) \) for \( g(x_i,t) \) too.

What is (120)? This is nothing but a system of ordinary differential equations in \( N-1 \) unknowns \( u_1(t),\ldots,u_{N-1}(t) \)! In other words, with aid of the finite difference approximation (119), we have reduced the single partial differential equation to a system of ODEs, which we know how to solve. In the literature, this strategy is called the method of lines.

We need to look into the initial and boundary conditions as well. The initial condition \( u(x,0)=I(x) \) translates to an initial condition for every unknown function \( u_i(t) \): \( u_i(0)=I(x_i) \), \( i=0,\ldots,N \). At the boundary \( x=0 \) we need an ODE in our ODE system, which must come from the boundary condition at this point. The boundary condition reads \( u(0,t)=s(t) \). We can derive an ODE from this equation by differentiating both sides: \( u_0'(t)=s'(t) \). The ODE system above cannot be used for \( u_0' \) since that equation involves some quantity \( u_{-1}' \) outside the domain. Instead, we use the equation \( u_0'(t)=s'(t) \) derived from the boundary condition. For this particular equation we also need to make sure the initial condition is \( u_0(0)=s(0) \) (otherwise nothing will happen: we get \( u=283 \) K forever).

We remark that a separate ODE for the (known) boundary condition \( u_0=s(t) \) is not strictly needed. We can just work with the ODE system for \( u_1,\ldots,u_{N} \), and in the ODE for \( u_0 \), replace \( u_0(t) \) by \( s(t) \). However, these authors prefer to have an ODE for every point value \( u_i \), \( i=0,\ldots,N \), which requires formulating the known boundary at \( x=0 \) as an ODE. The reason for including the boundary values in the ODE system is that the solution of the system is then the complete solution at all mesh points, which is convenient, since special treatment of the boundary values is then avoided.

The condition \( \partial u/\partial x=0 \) at \( x=L \) is a bit more complicated, but we can approximate the spatial derivative by a centered finite difference: $$ \left.\frac{\partial u}{\partial x}\right|_{i=N}\approx \frac{u_{N+1}-u_{N-1}}{2\Delta x} = 0\thinspace .$$ This approximation involves a fictitious point \( x_{N+1} \) outside the domain. A common trick is to use (120) for \( i=N \) and eliminate \( u_{N+1} \) by use of the discrete boundary condition (\( u_{N+1}=u_{N-1} \)): $$ \begin{equation} \frac{d u_N(t)}{d t} = \beta \frac{2u_{N-1}(t) - 2u_N(t)}{\Delta x^2} + g_N(t)\thinspace . \tag{121} \end{equation} $$ That is, we have a special version of (120) at the boundary \( i=N \).

What about simpler finite differences at the boundary?

Some reader may think that a smarter trick is to approximate the boundary condition \( \partial u/\partial x \) at \( x=L \) by a one-sided difference: $$ \left.\frac{\partial u}{\partial x}\right|_{i=N}\approx \frac{u_{N}-u_{N-1}}{\Delta x} = 0\thinspace .$$ This gives a simple equation \( u_N=u_{N-1} \) for the boundary value, and a corresponding ODE \( u_N'=u_{N-1}' \). However, this approximation has an error of order \( \Delta x \), while the centered approximation we used above has an error of order \( \Delta x^2 \). The finite difference approximation we used for the second-order derivative in the diffusion equation also has an error of order \( \Delta x^2 \). Thus, if we use the simpler one-sided difference above, it turns out that we reduce the overall accuracy of the method.

We are now in a position to summarize how we can approximate the partial differential equation problem (114)-(117) by a system of ordinary differential equations: $$ \begin{align} \frac{du_0}{dt} &= s'(t), \tag{122}\\ \frac{du_i}{dt} &= \frac{\beta}{\Delta x^2} (u_{i+1}(t) - 2u_i(t) + u_{i-1}(t)) + g_i(t),\quad i=1,\ldots,N-1, \tag{123}\\ \frac{du_N}{dt} &= \frac{2\beta}{\Delta x^2} (u_{N-1}(t) - u_N(t)) + g_N(t)\thinspace . \tag{124} \end{align} $$ The initial conditions are $$ \begin{align} u_0(0) &= s(0), \tag{125}\\ u_i(0) &= I(x_i),\quad i=1,\ldots,N\thinspace . \tag{126} \end{align} $$ We can apply any method for systems of ODEs to solve (122)-(124).

Construction of a test problem with known discrete solution

At this point, it is tempting to implement a real physical case and run it. However, partial differential equations constitute a non-trivial topic where mathematical and programming mistakes come easy. A better start is therefore to address a carefully designed test example where we can check that the method works. The most attractive examples for testing implementations are those without approximation errors, because we know exactly what numbers the program should produce. It turns out that solutions \( u(x,t) \) that are linear in time and in space can be exactly reproduced by most numerical methods for partial differential equations. A candidate solution might be $$ u(x,t) = (3t+2)(x-L)\thinspace .$$ Inserting this \( u \) in the governing equation gives $$ 3(x-L) = 0 + g(x,t) \quad\Rightarrow\quad g(x,t)= 3(x-L) \thinspace .$$ What about the boundary conditions? We realize that \( \partial u/\partial x = 3t+2 \) for \( x=L \), which breaks the assumption of \( \partial u/\partial x=0 \) at \( x=L \) in the formulation of the numerical method above. Moreover, \( u(0,t)=-L(3t+2) \), so we must set \( s(t)=-L(3t+2) \) and \( s'(t)=-3L \). Finally, the initial condition dictates \( I(x)=2(x-L) \), but recall that we must have \( u_0=s(0) \), and \( u_i=I(x_i) \), \( i=1,\ldots,N \): it is important that \( u_0 \) starts out at the right value dictated by \( s(t) \) in case \( I(0) \) is not equal this value.

First we need to generalize our method to handle \( \partial u/\partial x=\gamma \neq 0 \) at \( x=L \). We then have $$ \frac{u_{N+1}(t)- u_{N-1}(t)}{2\Delta x}= \gamma\quad\Rightarrow \quad u_{N+1} = u_{N-1} + 2\gamma \Delta x,$$ which inserted in (120) gives $$ \begin{equation} \frac{d u_N(t)}{d t} = \beta \frac{2u_{N-1}(t) + 2\gamma\Delta x - 2u_N(t)}{\Delta x^2} + g_N(t)\thinspace . \tag{127} \end{equation} $$

Implementation: Forward Euler method

In particular, we may use the Forward Euler method as implemented in the general function ode_FE in the module ode_system_FE from the section Programming the numerical method; the general case. The ode_FE function needs a specification of the right-hand side of the ODE system. This is a matter of translating (122), (123), and (127) to Python code (in file test_diffusion_pde_exact_linear.py):

def rhs(u, t):
    N = len(u) - 1
    rhs = zeros(N+1)
    rhs[0] = dsdt(t)
    for i in range(1, N):
        rhs[i] = (beta/dx**2)*(u[i+1] - 2*u[i] + u[i-1]) + \ 
                 g(x[i], t)
    rhs[N] = (beta/dx**2)*(2*u[N-1] + 2*dx*dudx(t) -
                           2*u[N]) + g(x[N], t)
    return rhs

def u_exact(x, t):
    return (3*t + 2)*(x - L)

def dudx(t):
    return (3*t + 2)

def s(t):
    return u_exact(0, t)

def dsdt(t):
    return 3*(-L)

def g(x, t):
    return 3*(x-L)
Note that dudx(t) is the function representing the \( \gamma \) parameter in (127). Also note that the rhs function relies on access to global variables beta, dx, L, and x, and global functions dsdt, g, and dudx.

We expect the solution to be correct regardless of \( N \) and \( \Delta t \), so we can choose a small \( N \), \( N=4 \), and \( \Delta t = 0.1 \). A test function with \( N=4 \) goes like

def test_diffusion_exact_linear():
    global beta, dx, L, x  # needed in rhs
    L = 1.5
    beta = 0.5
    N = 4
    x = linspace(0, L, N+1)
    dx = x[1] - x[0]
    u = zeros(N+1)

    U_0 = zeros(N+1)
    U_0[0] = s(0)
    U_0[1:] = u_exact(x[1:], 0)
    dt = 0.1
    print dt

    u, t = ode_FE(rhs, U_0, dt, T=1.2)

    tol = 1E-12
    for i in range(0, u.shape[0]):
        diff = abs(u_exact(x, t[i]) - u[i,:]).max()
        assert diff < tol, 'diff=%.16g' % diff
        print 'diff=%g at t=%g' % (diff, t[i])
With \( N=4 \) we reproduce the linear solution exactly. This brings confidence to the implementation, which is just what we need for attacking a real physical problem next.

Problems with reusing the rhs function.

The rhs function must take u and t as arguments, because that is required by the ode_FE function. What about the variables beta, dx, L, x, dsdt, g, and dudx that the rhs function needs? These are global in the solution we have presented so far. Unfortunately, this has an undesired side effect: we cannot import the rhs function in a new file, define dudx and dsdt in this new file and get the imported rhs to use these functions. The imported rhs will use the global variables, including functions, in its own module.

How can we find solutions to this problem? Technically, we must pack the extra data beta, dx, L, x, dsdt, g, and dudx with the rhs function, which requires more advanced programming considered beyond the scope of this text.

A class is the simplest construction for packing a function together with data, see the beginning of Chapter 7 in [18] for a detailed example on how classes can be used in such a context. Another solution in Python, and especially in computer languages supporting functional programming, is so called closures. They are also covered in Chapter 7 in the mentioned reference and behave in a magic way. The third solution is to allow an arbitrary set of arguments for rhs in a list to be transferred to ode_FE and then back to rhs. Appendix H.4 in [18] explains the technical details.

Application: heat conduction in a rod

Let us return to the case with heat conduction in a rod (114)-(117). Assume that the rod is 50 cm long and made of aluminum alloy 6082. The \( \beta \) parameter equals \( \kappa/(\varrho c) \), where \( \kappa \) is the heat conduction coefficient, \( \varrho \) is the density, and \( c \) is the heat capacity. We can find proper values for these physical quantities in the case of aluminum alloy 6082: \( \varrho = 2.7\cdot 10^3\hbox{ kg/m}^3 \), \( \kappa = 200\,\,\frac{\hbox{W}}{\hbox{mK}} \), \( c=900\,\,\frac{\hbox{J}}{\hbox{Kkg}} \). This results in \( \beta = \kappa/(\varrho c) = 8.2\cdot 10^{-5}\hbox{ m}^2/\hbox{s} \). Preliminary simulations show that we are close to a constant steady state temperature after 1 h, i.e., \( T=3600 \) s.

The rhs function from the previous section can be reused, only the functions s, dsdt, g, and dudx must be changed (see file rod_FE.py):

def dudx(t):
    return 0

def s(t):
    return 323

def dsdt(t):
    return 0

def g(x, t):
    return 0
Parameters can be set as

L = 0.5
beta = 8.2E-5
N = 40
x = linspace(0, L, N+1)
dx = x[1] - x[0]
u = zeros(N+1)

U_0 = zeros(N+1)
U_0[0] = s(0)
U_0[1:] = 283
Let us use \( \Delta t = 0.00034375 \). We can now call ode_FE and then make an animation on the screen to see how \( u(x,t) \) develops in time:

t0 = time.clock()
from ode_system_FE import ode_FE
u, t = ode_FE(rhs, U_0, dt, T=1*60*60)
t1 = time.clock()
print 'CPU time: %.1fs' % (t1 - t0)

# Make movie
import os
os.system('rm tmp_*.png')
import matplotlib.pyplot as plt
plt.ion()
y = u[0,:]
lines = plt.plot(x, y)
plt.axis([x[0], x[-1], 273, s(0)+10])
plt.xlabel('x')
plt.ylabel('u(x,t)')
counter = 0
# Plot each of the first 100 frames, then increase speed by 10x
change_speed = 100
for i in range(0, u.shape[0]):
    print t[i]
    plot = True if i <= change_speed else i % 10 == 0
    lines[0].set_ydata(u[i,:])
    if i > change_speed:
        plt.legend(['t=%.0f 10x' % t[i]])
    else:
        plt.legend(['t=%.0f' % t[i]])
    plt.draw()
    if plot:
        plt.savefig('tmp_%04d.png' % counter)
        counter += 1
    #time.sleep(0.2)

The plotting statements update the \( u(x,t) \) curve on the screen. In addition, we save a fraction of the plots to files tmp_0000.png, tmp_0001.png, tmp_0002.png, and so on. These plots can be combined to ordinary video files. A common tool is ffmpeg or its sister avconv.

These programs take the same type of command-line options. To make a Flash video movie.flv, run

Terminal> ffmpeg -i tmp_%04d.png -r 4 -vcodec flv movie.flv
The -i option specifies the naming of the plot files in printf syntax, and -r specifies the number of frames per second in the movie. On Mac, run ffmpeg instead of avconv with the same options. Other video formats, such as MP4, WebM, and Ogg can also be produced:

Terminal> ffmpeg -i tmp_%04d.png -r 4 -vcodec libx264   movie.mp4
Terminal> ffmpeg -i tmp_%04d.png -r 4 -vcodec libvpx    movie.webm
Terminal> ffmpeg -i tmp_%04d.png -r 4 -vcodec libtheora movie.ogg

The results of a simulation start out as in Figures 54 and 55. We see that the solution definitely looks wrong. The temperature is expected to be smooth, not having such a saw-tooth shape. Also, after some time (Figure 55), the temperature starts to increase much more than expected. We say that this solution is unstable, meaning that it does not display the same characteristics as the true, physical solution. Even though we tested the code carefully in the previous section, it does not seem to work for a physical application! How can that be?


Figure 54: Unstable simulation of the temperature in a rod.


Figure 55: Unstable simulation of the temperature in a rod.

The problem is that \( \Delta t \) is too large, making the solution unstable. It turns out that the Forward Euler time integration method puts a restriction on the size of \( \Delta t \). For the heat equation and the way we have discretized it, this restriction can be shown to be [20] $$ \begin{equation} \Delta t \leq \frac{\Delta x^2}{2\beta}\thinspace . \tag{128} \end{equation} $$ This is called a stability criterion. With the chosen parameters, (128) tells us that the upper limit is \( \Delta t=0.0003125 \), which is smaller than our choice above. Rerunning the case with a \( \Delta t \) equal to \( \Delta x^2/(2\beta) \), indeed shows a smooth evolution of \( u(x,t) \). Find the program rod_FE.py and run it to see an animation of the \( u(x,t) \) function on the screen.

Scaling and dimensionless quantities.

Our setting of parameters required finding three physical properties of a certain material. The time interval for simulation and the time step depend crucially on the values for \( \beta \) and \( L \), which can vary significantly from case to case. Often, we are more interested in how the shape of \( u(x,t) \) develops, than in the actual \( u \), \( x \), and \( t \) values for a specific material. We can then simplify the setting of physical parameters by scaling the problem.

Scaling means that we introduce dimensionless independent and dependent variables, here denoted by a bar: $$ \bar u = \frac{u-u^*}{u_c-u^*},\quad \bar x=\frac{x}{x_c},\quad \bar t = \frac{t}{t_c}, $$ where \( u_c \) is a characteristic size of the temperature, \( u^* \) is some reference temperature, while \( x_c \) and \( t_c \) are characteristic time and space scales. Here, it is natural to choose \( u^* \) as the initial condition, and set \( u_c \) to the stationary (end) temperature. Then \( \bar u\in [0,1] \), starting at 0 and ending at 1 as \( t\rightarrow\infty \). The length \( L \) is \( x_c \), while choosing \( t_c \) is more challenging, but one can argue for \( t_c = L^2/\beta \). The resulting equation for \( \bar u \) reads $$ \frac{\partial \bar u}{\partial \bar t} = \frac{\partial^2 \bar u}{\partial \bar x^2},\quad \bar x\in (0,1)\thinspace .$$ Note that in this equation, there are no physical parameters! In other words, we have found a model that is independent of the length of the rod and the material it is made of (!).

We can easily solve this equation with our program by setting \( \beta=1 \), \( L=1 \), \( I(x)=0 \), and \( s(t)=1 \). It turns out that the total simulation time (to "infinity") can be taken as 1.2. When we have the solution \( \bar u(\bar x,\bar t) \), the solution with dimension Kelvin, reflecting the true temperature in our medium, is given by $$ u(x,t) = u^* + (u_c-u^*)\bar u(x/L, t\beta/L^2)\thinspace .$$ Through this formula we can quickly generate the solutions for a rod made of aluminum, wood, or rubber - it is just a matter of plugging in the right \( \beta \) value.

Figure 56 shows four snapshots of the scaled (dimensionless) solution \( \bar (\bar x,\bar t) \).

The power of scaling is to reduce the number of physical parameters in a problem, and in the present case, we found one single problem that is independent of the material (\( \beta \)) and the geometry (\( L \)).


Figure 56: Snapshots of the dimensionless solution of a scaled problem.

Vectorization

Occasionally in this book, we show how to speed up code by replacing loops over arrays by vectorized expressions. The present problem involves a loop for computing the right-hand side:

for i in range(1, N):
    rhs[i] = (beta/dx**2)*(u[i+1] - 2*u[i] + u[i-1]) + g(x[i], t)
This loop can be replaced by a vectorized expression with the following reasoning. We want to set all the inner points at once: rhs[1:N-1] (this goes from index 1 up to, but not including, N). As the loop index i runs from 1 to N-1, the u[i+1] term will cover all the inner u values displaced one index to the right (compared to 1:N-1), i.e., u[2:N]. Similarly, u[i-1] corresponds to all inner u values displaced one index to the left: u[0:N-2]. Finally, u[i] has the same indices as rhs: u[1:N-1]. The vectorized loop can therefore be written in terms of slices:

rhs[1:N-1] = (beta/dx**2)*(u[2:N+1] - 2*u[1:N] + u[0:N-1]) +
           g(x[1:N], t)
This rewrite speeds up the code by about a factor of 10. A complete code is found in the file rod_FE_vec.py.

Using Odespy to solve the system of ODEs

Let us now show how to apply a general ODE package like Odespy (see the section Software for solving ODEs) to solve our diffusion problem. As long as we have defined a right-hand side function rhs this is very straightforward:

import odespy
solver = odespy.RKFehlberg(rhs)
solver.set_initial_condition(U_0)
T = 1.2
N_t = int(round(T/float(dt)))
time_points = linspace(0, T, N_t+1)
u, t = solver.solve(time_points)

# Check how many time steps are required by adaptive vs
# fixed-step methods
if hasattr(solver, 't_all'):
    print '# time steps:', len(solver.t_all)
else:
    print '# time steps:', len(t)

The very nice thing is that we can now easily experiment with many different integration methods. Trying out some simple ones first, like RK2 and RK4, quickly reveals that the time step limitation of the Forward Euler scheme also applies to these more sophisticated Runge-Kutta methods, but their accuracy is better. However, the Odespy package offers also adaptive methods. We can then specify a much larger time step in time_points, and the solver will figure out the appropriate step. Above we indicated how to use the adaptive Runge-Kutta-Fehlberg 4-5 solver. While the \( \Delta t \) corresponding to the Forward Euler method requires over 8000 steps for a simulation, we started the RKFehlberg method with 100 times this time step and in the end it required just slightly more than 2500 steps, using the default tolerance parameters. Lowering the tolerance did not save any significant amount of computational work. Figure 57 shows a comparison of the length of all the time steps for two values of the tolerance. We see that the influence of the tolerance is minor in this computational example, so it seems that the blow-up due to instability is what governs the time step size. The nice feature of this adaptive method is that we can just specify when we want the solution to be computed, and the method figures out on its own what time step that has to be used because of stability restrictions.


Figure 57: Time steps used by the Runge-Kutta-Fehlberg method: error tolerance \( 10^{-3} \) (left) and \( 10^{-6} \) (right).

We have seen how easy it is to apply sophisticated methods for ODEs to this PDE example. We shall take the use of Odespy one step further in the next section.

Implicit methods

A major problem with the stability criterion (128) is that the time step becomes very small if \( \Delta x \) is small. For example, halving \( \Delta x \) requires four times as many time steps and eight times the work. Now, with \( N=40 \), which is a reasonable resolution for the test problem above, the computations are very fast. What takes time, is the visualization on the screen, but for that purpose one can visualize only a subset of the time steps. However, there are occasions when you need to take larger time steps with the diffusion equation, especially if interest is in the long-term behavior as \( t\rightarrow\infty \). You must then turn to implicit methods for ODEs. These methods require the solutions of linear systems, if the underlying PDE is linear, and systems of nonlinear algebraic equations if the underlying PDE is non-linear.

The simplest implicit method is the Backward Euler scheme, which puts no restrictions on \( \Delta t \) for stability, but obviously, a large \( \Delta t \) leads to inaccurate results. The Backward Euler scheme for a scalar ODE \( u' = f(u,t) \) reads $$ \frac{u^{n+1} - u^{n}}{\Delta t} = f(u^{n+1}, t_{n+1})\thinspace .$$ This equation is to be solved for \( u^{n+1} \). If \( f \) is linear in \( u \), it is a linear equation, but if \( f \) is nonlinear in \( u \), one needs approximate methods for nonlinear equations (the chapter Solving nonlinear algebraic equations).

In our case, we have a system of linear ODEs (122)-(124). The Backward Euler scheme applied to each equation leads to $$ \begin{align} \frac{u_0^{n+1}-u_0^n}{\Delta t} &= s'(t_{n+1}), \tag{129}\\ \frac{u_i^{n+1} - u_i^{n}}{\Delta t} &= \frac{\beta}{\Delta x^2} (u_{i+1}^{n+1} - 2u_i^{n+1} + u_{i-1}^{n+1}) + g_i(t_{n+1}), \tag{130}\\ &\qquad\qquad \quad i=1,\ldots,N-1,\nonumber\\ \frac{u_N^{n+1} - u_N^{n}}{\Delta t} &= \frac{2\beta}{\Delta x^2} (u_{N-1}^{n+1} - u_N^{n+1}) + g_i(t_{n+1})\thinspace . \tag{131} \end{align} $$ This is a system of linear equations in the unknowns \( u_i^{n+1} \), \( i=0,\ldots,N \), which is easy to realize by writing out the equations for the case \( N=3 \), collecting all the unknown terms on the left-hand side and all the known terms on the right-hand side: $$ \begin{align} u_0^{n+1} &= u_0^n + \Delta t\,s'(t_{n+1}), \tag{132}\\ u_1^{n+1} - \Delta t \frac{\beta}{\Delta x^2} (u_{2}^{n+1} - 2u_1^{n+1} + u_{0}^{n+1}) &= u_1^{n} + \Delta t\,g_1(t_{n+1}), \tag{133}\\ u_2^{n+1} - \Delta t\frac{2\beta}{\Delta x^2} (u_{1}^{n+1} - u_2^{n+1}) &= u_2^{n} + \Delta t\,g_2(t_{n+1})\thinspace . \tag{134} \end{align} $$

A system of linear equations like this, is usually written on matrix form \( Au=b \), where \( A \) is a coefficient matrix, \( u=(u_0^{n+1},\ldots,n_N^{n+1}) \) is the vector of unknowns, and \( b \) is a vector of known values. The coefficient matrix for the case (132)-(134) becomes $$ A = \left(\begin{array}{ccc} 1 & 0 & 0\\ -\Delta t \frac{\beta}{\Delta x^2} & 1 + 2\Delta t \frac{\beta}{\Delta x^2} & - \Delta t \frac{\beta}{\Delta x^2}\\ 0 & - \Delta t\frac{2\beta}{\Delta x^2} & 1 + \Delta t\frac{2\beta}{\Delta x^2} \end{array}\right) $$ In the general case (129)-(131), the coefficient matrix is an \( (N+1)\times(N+1) \) matrix with zero entries, except for $$ \begin{align} A_{1,1} &= 1 \tag{135}\\ A_{i,i-1} &= -\Delta t \frac{\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{136}\\ A_{i,i+1} &= -\Delta t \frac{\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{137}\\ A_{i,i} &= 1 + 2\Delta t \frac{\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{138}\\ A_{N,N-1} & = - \Delta t\frac{2\beta}{\Delta x^2} \tag{139}\\ A_{N,N} &= 1 + \Delta t\frac{2\beta}{\Delta x^2} \tag{140} \end{align} $$

If we want to apply general methods for systems of ODEs on the form \( u'=f(u,t) \), we can assume a linear \( f(u,t)=Ku \). The coefficient matrix \( K \) is found from the right-hand side of (129)-(131) to be $$ \begin{align} K_{1,1} &= 0 \tag{141}\\ K_{i,i-1} &= \frac{\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{142}\\ K_{i,i+1} &= \frac{\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{143}\\ K_{i,i} &= -\frac{2\beta}{\Delta x^2},\quad i=2,\ldots,N-1 \tag{144}\\ K_{N,N-1} & = \frac{2\beta}{\Delta x^2} \tag{145}\\ K_{N,N} &= -\frac{2\beta}{\Delta x^2} \tag{146} \end{align} $$ We see that \( A=I-\Delta t\,K \).

To implement the Backward Euler scheme, we can either fill a matrix and call a linear solver, or we can apply Odespy. We follow the latter strategy. Implicit methods in Odespy need the \( K \) matrix above, given as an argument jac (Jacobian of \( f \)) in the call to odespy.BackwardEuler. Here is the Python code for the right-hand side of the ODE system (rhs) and the \( K \) matrix (K) as well as statements for initializing and running the Odespy solver BackwardEuler (in the file rod_BE.py):

def rhs(u, t):
    N = len(u) - 1
    rhs = zeros(N+1)
    rhs[0] = dsdt(t)
    for i in range(1, N):
        rhs[i] = (beta/dx**2)*(u[i+1] - 2*u[i] + u[i-1]) + \ 
                 g(x[i], t)
    rhs[N] = (beta/dx**2)*(2*u[i-1] + 2*dx*dudx(t) -
                           2*u[i]) + g(x[N], t)
    return rhs

def K(u, t):
    N = len(u) - 1
    K = zeros((N+1,N+1))
    K[0,0] = 0
    for i in range(1, N):
        K[i,i-1] = beta/dx**2
        K[i,i] = -2*beta/dx**2
        K[i,i+1] = beta/dx**2
    K[N,N-1] = (beta/dx**2)*2
    K[N,N] = (beta/dx**2)*(-2)
    return K

import odespy
solver = odespy.BackwardEuler(rhs, f_is_linear=True, jac=K)
solver = odespy.ThetaRule(rhs, f_is_linear=True, jac=K, theta=0.5)
solver.set_initial_condition(U_0)
T = 1*60*60
N_t = int(round(T/float(dt)))
time_points = linspace(0, T, N_t+1)
u, t = solver.solve(time_points)
The file rod_BE.py has all the details and shows a movie of the solution. We can run it with any \( \Delta t \) we want, its size just impacts the accuracy of the first steps.

Odespy solvers apply dense matrices!

Looking at the entries of the \( K \) matrix, we realize that there are at maximum three entries different from zero in each row. Therefore, most of the entries are zeroes. The Odespy solvers expect dense square matrices as input, here with \( (N+1)\times(N+1) \) elements. When solving the linear systems, a lot of storage and work are spent on the zero entries in the matrix. It would be much more efficient to store the matrix as a tridiagonal matrix and apply a specialized Gaussian elimination solver for tridiagonal systems. Actually, this reduces the work from the order \( N^3 \) to the order \( N \).

In one-dimensional diffusion problems, the savings of using a tridiagonal matrix are modest in practice, since the matrices are very small anyway. In two- and three-dimensional PDE problems, however, one cannot afford dense square matrices. Rather, one must resort to more efficient storage formats and algorithms tailored to such formats, but this is beyond the scope of the present text.

Exercises

Exercise 5.1: Simulate a diffusion equation by hand

Consider the problem given by (122), (123) and (127). Set \( N=2 \) and compute \( u_i^0 \), \( u_i^1 \) and \( u_i^2 \) by hand for \( i=0,1,2 \). Use these values to construct a test function for checking that the implementation is correct. Copy useful functions from test_diffusion_pde_exact_linear.py and make a new test function test_diffusion_hand_calculation.

Filename: test_rod_hand_calculations.py.

Exercise 5.2: Compute temperature variations in the ground

The surface temperature at the ground shows daily and seasonal oscillations. When the temperature rises at the surface, heat is propagated into the ground, and the coefficient \( \beta \) in the diffusion equation determines how fast this propagation is. It takes some time before the temperature rises down in the ground. At the surface, the temperature has then fallen. We are interested in how the temperature varies down in the ground because of temperature oscillations on the surface.

Assuming homogeneous horizontal properties of the ground, at least locally, and no variations of the temperature at the surface at a fixed point of time, we can neglect the horizontal variations of the temperature. Then a one-dimensional diffusion equation governs the heat propagation along a vertical axis called \( x \). The surface corresponds to \( x=0 \) and the \( x \) axis point downwards into the ground. There is no source term in the equation (actually, if rocks in the ground are radioactive, they emit heat and that can be modeled by a source term, but this effect is neglected here).

At some depth \( x=L \) we assume that the heat changes in \( x \) vanish, so \( \partial u/\partial x=0 \) is an appropriate boundary condition at \( x=L \). We assume a simple sinusoidal temperature variation at the surface: $$ u(0,t) = T_0 + T_a\sin\left(\frac{2\pi}{P}t\right),$$ where \( P \) is the period, taken here as 24 hours (\( 24\cdot 60\cdot 60 \) s). The \( \beta \) coefficient may be set to \( 10^{-6}\hbox{ m}^2/\hbox{s} \). Time is then measured in seconds. Set appropriate values for \( T_0 \) ad \( T_a \).

a) Show that the present problem has an analytical solution of the form $$ u(x,t) = A + Be^{-rx}\sin(\omega t - rx),$$ for appropriate values of \( A \), \( B \), \( r \), and \( \omega \).

b) Solve this heat propagation problem numerically for some days and animate the temperature. You may use the Forward Euler method in time. Plot both the numerical and analytical solution. As initial condition for the numerical solution, use the exact solution during program development, and when the curves coincide in the animation for all times, your implementation works, and you can then switch to a constant initial condition: \( u(x,0)=T_0 \). For this latter initial condition, how many periods of oscillations are necessary before there is a good (visual) match between the numerical and exact solution (despite differences at \( t=0 \))?

Filename: ground_temp.py.

Exercise 5.3: Compare implicit methods

An equally stable, but more accurate method than the Backward Euler scheme, is the so-called 2-step backward scheme, which for an ODE \( u'=f(u,t) \) can be expressed by $$ \frac{3u^{n+1} - 4u^{n} + u^{n-1}}{2\Delta t} = f(u^{n+1},t_{n+1}) \thinspace .$$ The Odespy package offers this method as odespy.Backward2Step. The purpose of this exercise is to compare three methods and animate the three solutions:

  1. The Backward Euler method with \( \Delta t =0.001 \)
  2. The backward 2-step method with \( \Delta t =0.001 \)
  3. The backward 2-step method with \( \Delta t =0.01 \)
Choose the model problem from the section Application: heat conduction in a rod.

Filename: rod_BE_vs_B2Step.py.

Exercise 5.4: Explore adaptive and implicit methods

We consider the same problem as in Exercise 5.2: Compute temperature variations in the ground. Now we want to explore the use of adaptive and implicit methods from Odespy to see if they are more efficient than the Forward Euler method. Assume that you want the accuracy provided by the Forward Euler method with its maximum \( \Delta t \) value. Since there exists an analytical solution, you can compute an error measure that summarizes the error in space and time over the whole simulation: $$ E = \sqrt{\Delta x\Delta t\sum_{i}\sum_n (U_i^n - u_i^n)^2}\thinspace .$$ Here, \( U_i^n \) is the exact solution. Use the Odespy package to run the following implicit and adaptive solvers:

  1. BackwardEuler
  2. Backward2Step
  3. RKFehlberg
Experiment to see if you can use larger time steps than what is required by the Forward Euler method and get solutions with the same order of accuracy.

Hint. To avoid oscillations in the solutions when using the RKFehlberg method, the rtol and atol parameters to RKFFehlberg must be set no larger than 0.001 and 0.0001, respectively. You can print out solver_RKF.t_all to see all the time steps used by the RKFehlberg solver (if solver is the RKFehlberg object). You can then compare the number of time steps with what is required by the other methods.

Filename: ground_temp_adaptive.py.

Exercise 5.5: Investigate the \( \theta \) rule

a) The Crank-Nicolson method for ODEs is very popular when combined with diffusion equations. For a linear ODE \( u'=au \) it reads $$ \frac{u^{n+1}-u^n}{\Delta t} = \frac{1}{2}(au^{n} + au^{n+1}) \thinspace .$$

Apply the Crank-Nicolson method in time to the ODE system for a one-dimensional diffusion equation. Identify the linear system to be solved.

b) The Backward Euler, Forward Euler, and Crank-Nicolson methods can be given a unified implementation. For a linear ODE \( u'=au \) this formulation is known as the \( \theta \) rule: $$ \frac{u^{n+1}-u^n}{\Delta t} = (1-\theta)au^{n} + \theta au^{n+1} \thinspace .$$ For \( \theta =0 \) we recover the Forward Euler method, \( \theta=1 \) gives the Backward Euler scheme, and \( \theta=1/2 \) corresponds to the Crank-Nicolson method. The approximation error in the \( \theta \) rule is proportional to \( \Delta t \), except for \( \theta =1/2 \) where it is proportional to \( \Delta t^2 \). For \( \theta \geq 1/2 \) the method is stable for all \( \Delta t \).

Apply the \( \theta \) rule to the ODE system for a one-dimensional diffusion equation. Identify the linear system to be solved.

c) Implement the \( \theta \) rule with aid of the Odespy package. The relevant object name is ThetaRule:

solver = odespy.ThetaRule(rhs, f_is_linear=True, jac=K, theta=0.5)

d) Consider the physical application from the section Application: heat conduction in a rod. Run this case with the \( \theta \) rule and \( \theta =1/2 \) for the following values of \( \Delta t \): 0.001, 0.01, 0.05. Report what you see.

Filename: rod_ThetaRule.py.

Remarks

Despite the fact that the Crank-Nicolson method, or the \( \theta \) rule with \( \theta=1/2 \), is theoretically more accurate than the Backward Euler and Forward Euler schemes, it may exhibit non-physical oscillations as in the present example if the solution is very steep. The oscillations are damped in time, and decreases with decreasing \( \Delta t \). To avoid oscillations one must have \( \Delta t \) at maximum twice the stability limit of the Forward Euler method. This is one reason why the Backward Euler method (or a 2-step backward scheme, see Exercise 5.3: Compare implicit methods) are popular for diffusion equations with abrupt initial conditions.

Exercise 5.6: Compute the diffusion of a Gaussian peak

Solve the following diffusion problem: $$ \begin{align} \frac{\partial u}{\partial t} &= \beta\frac{\partial^2 u}{\partial x^2}, & x\in (-1,1),\ t\in (0,T] \tag{147}\\ u(x,0) &= \frac{1}{\sqrt{2\pi}\sigma}\exp{\left(-\frac{x^2}{2\sigma^2}\right)}, & x\in [-1,1], \tag{148}\\ \frac{\partial}{\partial x}u(-1,t) & = 0 & t\in (0,T], \tag{149}\\ \frac{\partial}{\partial x}u(1,t) & = 0 & t\in (0,T]\thinspace . \tag{150} \end{align} $$ The initial condition is the famous and widely used Gaussian function with standard deviation (or "width") \( \sigma \), which is here taken to be small, \( \sigma = 0.01 \), such that the initial condition is a peak. This peak will then diffuse and become lower and wider. Compute \( u(x,t) \) until \( u \) becomes approximately constant over the domain.

Filename: gaussian_diffusion.py.

Remarks

Running the simulation with \( \sigma =0.2 \) results in a constant solution \( u\approx 1 \) as \( t\rightarrow\infty \), while one might expect from "physics of diffusion" that the solution should approach zero. The reason is that we apply Neumann conditions as boundary conditions. One can then easily show that the area under the \( u \) curve remains constant. Integrating the PDE gives $$ \int_{-1}^1 \frac{\partial u}{\partial t}dx = \beta \int_{-1}^1 \frac{\partial d^2 u}{\partial x^2}dx\thinspace . $$ Using the Gauss divergence theorem on the integral on the right-hand and moving the time-derivative outside the integral on the left-hand side results in $$ \frac{\partial}{\partial t} \int_{-1}^1 u(x,t) dx = \beta \left[\frac{\partial du}{\partial x}\right]_{-1}^1 = 0. $$ (Recall that \( \partial u/\partial x=0 \) at the end points.) The result means that \( \int_{-1}^1 udx \) remains constant during the simulation. Giving the PDE an interpretation in terms of heat conduction can easily explain the result: with Neumann conditions no heat can escape from the domain so the initial heat will just be evenly distributed, but not leak out, so the temperature cannot go to zero (or the scaled and translated temperature \( u \), to be precise). The area under the initial condition is 1, so with a sufficiently fine mesh, \( u\rightarrow 1 \), regardless of \( \sigma \).

Exercise 5.7: Vectorize a function for computing the area of a polygon

Vectorize the implementation of the function for computing the area of a polygon in Exercise 2.5: Area of a polygon. Make a test function that compares the scalar implementation in Exercise 2.5: Area of a polygon and the new vectorized implementation for the test cases used in Exercise 2.5: Area of a polygon.

Hint. Notice that the formula \( x_1y_2+x_2y_3 + \cdots + x_{n-1}y_n = \sum_{i=0}^{n-1}x_iy_{i+1} \) is the dot product of two vectors, x[:-1] and y[1:], which can be computed as numpy.dot(x[:-1], y[1:]), or more explicitly as numpy.sum(x[:-1]*y[1:]).

Filename: polyarea_vec.py.

Exercise 5.8: Explore symmetry

One can observe (and also mathematically prove) that the solution \( u(x,t) \) of the problem in Exercise 5.6: Compute the diffusion of a Gaussian peak is symmetric around \( x=0 \): \( u(-x,t) = u(x,t) \). In such a case, we can split the domain in two and compute \( u \) in only one half, \( [-1,0] \) or \( [0,1] \). At the symmetry line \( x=0 \) we have the symmetry boundary condition \( \partial u/\partial x=0 \). Reformulate the problem in Exercise 5.6: Compute the diffusion of a Gaussian peak such that we compute only for \( x\in [0,1] \). Display the solution and observe that it equals the right part of the solution in Exercise 5.6: Compute the diffusion of a Gaussian peak.

Filename: symmetric_gaussian_diffusion.py.

Remarks

In 2D and 3D problems, where the CPU time to compute a solution of PDE can be hours and days, it is very important to utilize symmetry as we do above to reduce the size of the problem.

Also note the remarks in Exercise 5.6: Compute the diffusion of a Gaussian peak about the constant area under the \( u(x,t) \) curve: here, the area is 0.5 and \( u\rightarrow 0.5 \) as \( t\rightarrow 0.5 \) (if the mesh is sufficiently fine - one will get convergence to smaller values for small \( \sigma \) if the mesh is not fine enough to properly resolve a thin-shaped initial condition).

Exercise 5.9: Compute solutions as \( t\rightarrow\infty \)

Many diffusion problems reach a stationary time-independent solution as \( t\rightarrow\infty \). The model problem from the section Application: heat conduction in a rod is one example where \( u(x,t)=s(t)=\hbox{const} \) for \( t\rightarrow\infty \). When \( u \) does not depend on time, the diffusion equation reduces to $$ -\beta u''(x) = f(x),$$ in one dimension, and $$ -\beta \nabla^2 u = f(x),$$ in 2D and 3D. This is the famous Poisson equation, or if \( f=0 \), it is known as the Laplace equation. In this limit \( t\rightarrow\infty \), there is no need for an initial condition, but the boundary conditions are the same as for the diffusion equation.

We now consider a one-dimensional problem $$ \begin{equation} -u''(x) = 0,\ x\in (0,L),\quad u(0)=C, \ u'(L)=0, \tag{151} \end{equation} $$ which is known as a two-point boundary value problem. This is nothing but the stationary limit of the diffusion problem in the section Application: heat conduction in a rod. How can we solve such a stationary problem (151)? The simplest strategy, when we already have a solver for the corresponding time-dependent problem, is to use that solver and simulate until \( t\rightarrow\infty \), which in practice means that \( u(x,t) \) no longer changes in time (within some tolerance).

A nice feature of implicit methods like the Backward Euler scheme is that one can take one very long time step to "infinity" and produce the solution of (151).

a) Let (151) be valid at mesh points \( x_i \) in space, discretize \( u'' \) by a finite difference, and set up a system of equations for the point values \( u_i \),$i =0,\ldots,N$, where \( u_i \) is the approximation at mesh point \( x_i \).

b) Show that if \( \Delta t\rightarrow\infty \) in (129) - (131), it leads to the same equations as in a).

c) Demonstrate, by running a program, that you can take one large time step with the Backward Euler scheme and compute the solution of (151). The solution is very boring since it is constant: \( u(x)=C \).

Filename: rod_stationary.py.

Remarks

If the interest is in the stationary limit of a diffusion equation, one can either solve the associated Laplace or Poisson equation directly, or use a Backward Euler scheme for the time-dependent diffusion equation with a very long time step. Using a Forward Euler scheme with small time steps is typically inappropriate in such situations because the solution changes more and more slowly, but the time step must still be kept small, and it takes "forever" to approach the stationary state. This is yet another example why one needs implicit methods like the Backward Euler scheme.

Exercise 5.10: Solve a two-point boundary value problem

Solve the following two-point boundary-value problem $$ u''(x) = 2,\ x\in (0,1),\quad u(0)=0,\ u(1)=1\thinspace .$$

Hint. Do Exercise 5.9: Compute solutions as \( t\rightarrow\infty \). Modify the boundary condition in the code so it incorporates a known value for \( u(1) \).

Filename: 2ptBVP.py.