The 1D diffusion equation

The famous diffusion equation, also known as the heat equation, reads

ut=α2ux2,

where u(x,t) is the unknown function to be solved for, x is a coordinate in space, and t is time. The coefficient α is the diffusion coefficient and determines how fast u changes in time. A quick short form for the diffusion equation is ut=αuxx.

Compared to the wave equation, utt=c2uxx, which looks very similar, but the diffusion equation features solutions that are very different from those of the wave equation. Also, the diffusion equation makes quite different demands to the numerical methods.

Typical diffusion problems may experience rapid change in the very beginning, but then the evolution of u becomes slower and slower. The solution is usually very smooth, and after some time, one cannot recognize the initial shape of u. This is in sharp contrast to solutions of the wave equation where the initial shape is preserved - the solution is basically a moving initial condition. The standard wave equation utt=c2uxx has solutions that propagates with speed c forever, without changing shape, while the diffusion equation converges to a stationary solution ˉu(x) as t. In this limit, ut=0, and ˉu is governed by ˉu(x)=0. This stationary limit of the diffusion equation is called the Laplace equation and arises in a very wide range of applications throughout the sciences.

It is possible to solve for u(x,t) using a explicit scheme, but the time step restrictions soon become much less favorable than for an explicit scheme for the wave equation. And of more importance, since the solution u of the diffusion equation is very smooth and changes slowly, small time steps are not convenient and not required by accuracy as the diffusion process converges to a stationary state.

The initial-boundary value problem for 1D diffusion

To obtain a unique solution of the diffusion equation, or equivalently, to apply numerical methods, we need initial and boundary conditions. The diffusion equation goes with one initial condition u(x,0)=I(x), where I is a prescribed function. One boundary condition is required at each point on the boundary, which in 1D means that u must be known, ux must be known, or some combination of them.

We shall start with the simplest boundary condition: u=0. The complete initial-boundary value diffusion problem in one space dimension can then be specified as

ut=α2ux2,x(0,L), t(0,T]
u(x,0)=I(x),x[0,L]
\begin{split}\tag{3} u(0,t)  = 0, \quad  t>0,\end{split}
\begin{split}\tag{4} u(L,t)  = 0, \quad  t>0{\thinspace .}\end{split}

Equation (1) is known as a one-dimensional diffusion equation, also often referred to as a heat equation. With only a first-order derivative in time, only one initial condition is needed, while the second-order derivative in space leads to a demand for two boundary conditions. The parameter α must be given and is referred to as the diffusion coefficient.

Diffusion equations like (1) have a wide range of applications throughout physical, biological, and financial sciences. One of the most common applications is propagation of heat, where u(x,t) represents the temperature of some substance at point x and time t.

Forward Euler scheme

The first step in the discretization procedure is to replace the domain [0,L]×[0,T] by a set of mesh points. Here we apply equally spaced mesh points

xi=iΔx,i=0,,Nx,

and

tn=nΔt,n=0,,Nt.

Moreover, uni denotes the mesh function that approximates u(xi,tn) for i=0,,Nx and n=0,,Nt. Requiring the PDE (1) to be fulfilled at a mesh point (xi,tn) leads to the equation

tu(xi,tn)=α2x2u(xi,tn),

The next step is to replace the derivatives by finite difference approximations. The computationally simplest method arises from using a forward difference in time and a central difference in space:

[D+tu=αDxDxu]ni.

Written out,

un+1iuniΔt=αuni+12uni+uni1Δx2.

We have turned the PDE into algebraic equations, also often called discrete equations. The key property of the equations is that they are algebraic, which makes them easy to solve. As usual, we anticipate that uni is already computed such that un+1i is the only unknown in (7). Solving with respect to this unknown is easy:

un+1i=uni+F(uni+12uni+uni1).

F is the key parameter in the discrete diffusion equation

Note that F is a dimensionless number that lumps the key physical parameter in the problem, α, and the discretization parameters Δx and Δt into a single parameter. All the properties of the numerical method are critically dependent upon the value of F (see the section Analysis of schemes for the diffusion equation for details).

The computational algorithm then becomes

  1. compute $u^0_i=I(x_i)$for i=0,,Nx
  2. for n=0,1,,Nt:
  1. apply (8) for all the internal spatial points i=1,,Nx1
  2. set the boundary values un+1i=0 for i=0 and i=Nx

The algorithm is compactly fully specified in Python:

x = linspace(0, L, Nx+1)    # mesh points in space
dx = x[1] - x[0]
t = linspace(0, T, Nt+1)    # mesh points in time
dt = t[1] - t[0]
F = a*dt/dx**2
u   = zeros(Nx+1)           # unknown u at new time level
u_1 = zeros(Nx+1)           # u at the previous time level

# Set initial condition u(x,0) = I(x)
for i in range(0, Nx+1):
    u_1[i] = I(x[i])

for n in range(0, Nt):
    # Compute u at inner mesh points
    for i in range(1, Nx):
        u[i] = u_1[i] + F*(u_1[i-1] - 2*u_1[i] + u_1[i+1])

    # Insert boundary conditions
    u[0] = 0;  u[Nx] = 0

    # Update u_1 before next step
    u_1[:]= u

The program diffu1D_u0.py contains a function solver_FE for solving the 1D diffusion equation with u=0 on the boundary. The functions plug and gaussian runs the case with I(x) as a discontinuous plug or a smooth Gaussian function, respectively. Experiments with these two functions reveal some important observations:

  • The Forward Euler scheme leads to growing solutions if F>12.
  • I(x) as a discontinuous plug leads to a saw tooth-like noise for F=12, see movie, which is absent for F14, see movie.
  • The smooth Gaussian initial function leads to a smooth solution, see movie for F=12.

Backward Euler scheme

We now apply a backward difference in time in (5), but the same central difference in space:

[Dtu=DxDxu]ni,

which written out reads

uniun1iΔt=αuni+12uni+uni1Δx2.

Now we assume un1i is computed, but all quantities at the “new” time level n are unknown. This time it is not possible to solve with respect to uni because this value couples to its neighbors in space, uni1 and uni+1, which are also unknown. Let us examine this fact for the case when Nx=3. Equation (10) written for i=1,,Nx1=1,2 becomes

un1un11Δt=αun22un1+un0Δx2
un2un12Δt=αun32un2+un1Δx2

The boundary values un0 and un3 are known as zero. Collecting the unknown new values un1 and un2 on the left-hand side gives

(1+2F)un1Fun2=un11,
Fun1+(1+2F)un2=un12.

This is a coupled 2×2 system of algebraic equations for the unknowns un1 and un2. The equivalent matrix form is

(1+2FFF1+2F)(un1un2)=(un11un12)

Implicit vs. explicit methods

Discretization methods that lead to a coupled system of equations for the unknown function at a new time level are said to be implicit methods. The counterpart, explicit methods, refers to discretization methods where there is a simple explicit formula for the values of the unknown function at each of the spatial mesh points at the new time level. From an implementational point of view, implicit methods are more comprehensive to code since they require the solution of coupled equations, i.e., a matrix system, at each time level.

In the general case, (10) gives rise to a coupled (Nx1)×(Nx1) system of algebraic equations for all the unknown uni at the interior spatial points i=1,,Nx1. Collecting the unknowns on the left-hand side, (10) can be written

Funi1+(1+2F)uniFuni+1=un1i1,

for i=1,,Nx1. Here, we have introduced the mesh Fourier number

F=αΔtΔx2.

One can either view these equations as a system for where the uni values at the internal mesh points, i=1,,Nx1, are unknown, or we may append the boundary values un0 and unNx to the system. In the latter case, all uni for i=0,,Nx are unknown and we must add the boundary equations to the Nx1 equations in (15):

un0=0,
unNx=0.

A coupled system of algebraic equations can be written on matrix form, and this is important if we want to call up ready-made software for solving the system. The equations (15) and (17)(18) correspond to the matrix equation

AU=b

where U=(un0,,unNx), and the matrix A has the following structure:

\begin{split}\tag{19} A =     \left(     \begin{array}{cccccccccc}     A_{0,0} & A_{0,1} & 0     &\cdots &     \cdots & \cdots & \cdots &     \cdots & 0 \\     A_{1,0} & A_{1,1} & 0 & \ddots &   & &  & &  \vdots \\     0 & A_{2,1} & A_{2,2} & A_{2,3} &     \ddots & &  &  & \vdots \\     \vdots & \ddots &  & \ddots & \ddots & 0 &  & & \vdots \\     \vdots &  & \ddots & \ddots & \ddots & \ddots & \ddots & & \vdots \\     \vdots & &  & 0 & A_{i,i-1} & A_{i,i} & A_{i,i+1} & \ddots & \vdots \\     \vdots & & &  & \ddots & \ddots & \ddots &\ddots  & 0 \\     \vdots & & & &  &\ddots  & \ddots &\ddots  & A_{N_x-1,N_x} \\     0 &\cdots & \cdots &\cdots & \cdots & \cdots  & 0 & A_{N_x,N_x-1} & A_{N_x,N_x}     \end{array}     \right)\end{split}

The nonzero elements are given by

Ai,i1=F
Ai,i=1+2F
Ai,i+1=F

for the equations for internal points, i=1,,Nx1. The equations for the boundary points correspond to

A0,0=1,
A0,1=0,
ANx,Nx1=0,
ANx,Nx=1.

The right-hand side b is written as

\begin{split}\tag{27} b = \left(\begin{array}{c}     b_0\\     b_1\\     \vdots\\     b_i\\     \vdots\\     b_{N_x}     \end{array}\right)\end{split}

with

b0=0,
bi=un1i,i=1,,Nx1,
bNx=0.

We observe that the matrix A contains quantities that do not change in time. Therefore, A can be formed once and for all before we enter the recursive formulas for the time evolution. The right-hand side b, however, must be updated at each time step. This leads to the following computational algorithm, here sketched with Python code:

x = linspace(0, L, Nx+1)   # mesh points in space
dx = x[1] - x[0]
t = linspace(0, T, N+1)    # mesh points in time
u   = zeros(Nx+1)          # unknown u at new time level
u_1 = zeros(Nx+1)          # u at the previous time level

# Data structures for the linear system
A = zeros((Nx+1, Nx+1))
b = zeros(Nx+1)

for i in range(1, Nx):
    A[i,i-1] = -F
    A[i,i+1] = -F
    A[i,i] = 1 + 2*F
A[0,0] = A[Nx,Nx] = 1

# Set initial condition u(x,0) = I(x)
for i in range(0, Nx+1):
    u_1[i] = I(x[i])

import scipy.linalg

for n in range(0, Nt):
    # Compute b and solve linear system
    for i in range(1, Nx):
        b[i] = -u_1[i]
    b[0] = b[Nx] = 0
    u[:] = scipy.linalg.solve(A, b)

    # Update u_1 before next step
    u_1[:] = u

Sparse matrix implementation

We have seen from (19) that the matrix A is tridiagonal. The code segment above used a full, dense matrix representation of A, which stores a lot of values we know are zero beforehand, and worse, the solution algorithm computes with all these zeros. With Nx+1 unknowns, the work by the solution algorithm is 13(Nx+1)3 and the storage requirements (Nx+1)2. By utilizing the fact that A is tridiagonal and employing corresponding software tools, the work and storage demands can be proportional to Nx only.

The key idea is to apply a data structure for a tridiagonal or sparse matrix. The scipy.sparse package has relevant utilities. For example, we can store the nonzero diagonals of a matrix. The package also has linear system solvers that operate on sparse matrix data structures. The code below illustrates how we can store only the main diagonal and the upper and lower diagonals.

# Representation of sparse matrix and right-hand side
main  = zeros(Nx+1)
lower = zeros(Nx-1)
upper = zeros(Nx-1)
b     = zeros(Nx+1)

# Precompute sparse matrix
main[:] = 1 + 2*F
lower[:] = -F  #1
upper[:] = -F  #1
# Insert boundary conditions
main[0] = 1
main[Nx] = 1

A = scipy.sparse.diags(
    diagonals=[main, lower, upper],
    offsets=[0, -1, 1], shape=(Nx+1, Nx+1),
    format='csr')
print A.todense()  # Check that A is correct

# Set initial condition
for i in range(0,Nx+1):
    u_1[i] = I(x[i])

for n in range(0, Nt):
    b = u_1
    b[0] = b[-1] = 0.0  # boundary conditions
    u[:] = scipy.sparse.linalg.spsolve(A, b)
    u_1[:] = u

The scipy.sparse.linalg.spsolve function utilizes the sparse storage structure of A and performs in this case a very efficient Gaussian elimination solve.

The program diffu1D_u0.py contains a function solver_BE, which implements the Backward Euler scheme sketched above. As mentioned in the section Forward Euler scheme, the functions plug and gaussian runs the case with I(x) as a discontinuous plug or a smooth Gaussian function. All experiments point to two characteristic features of the Backward Euler scheme: 1) it is always stable, and 2) it always gives a smooth, decaying solution.

Crank-Nicolson scheme

The idea in the Crank-Nicolson scheme is to apply centered differences in space and time, combined with an average in time. We demand the PDE to be fulfilled at the spatial mesh points, but in between the points in the time mesh:

tu(xi,tn+12)=α2x2u(xi,tn+12).

for i=1,,Nx1 and n=0,,Nt1.

With centered differences in space and time, we get

[Dtu=αDxDxu]n+12i.

On the right-hand side we get an expression

1Δx2(un+12i12un+12i+un+12i+1).

This expression is problematic since un+12i is not one of the unknown we compute. A possibility is to replace un+12i by an arithmetic average:

un+12i12(uni+un+1i).

In the compact notation, we can use the arithmetic average notation ¯ut:

[Dtu=αDxDx¯ut]n+12i.

After writing out the differences and average, multiplying by Δt, and collecting all unknown terms on the left-hand side, we get

un+1i12F(un+1i12un+1i+un+1i+1)=uni+12F(uni12uni+uni+1).

Also here, as in the Backward Euler scheme, the new unknowns un+1i1, un+1i, and un+1i+1 are coupled in a linear system AU=b, where A has the same structure as in (19), but with slightly different entries:

Ai,i1=12F
Ai,i=12+F
Ai,i+1=12F

for the equations for internal points, i=1,,Nx1. The equations for the boundary points correspond to

A0,0=1,
A0,1=0,
ANx,Nx1=0,
ANx,Nx=1.

The right-hand side b has entries

b0=0,
bi=un1i,i=1,,Nx1,
bNx=0.

The θ rule

For the equation

ut=G(u),

where G(u) is some a spatial differential operator, the θ-rule looks like

un+1iuniΔt=θG(un+1i)+(1θ)G(uni).

The important feature of this time discretization scheme is that we can implement one formula and then generate a family of well-known and widely used schemes:

  • θ=0 gives the Forward Euler scheme in time
  • θ=1 gives the Backward Euler scheme in time
  • θ=12 gives the Crank-Nicolson scheme in time

Applied to the 1D diffusion problem, the θ-rule gives

un+1iuniΔt=α(θun+1i+12un+1i+un+1i1Δx2+(1θ)uni+12uni+uni1Δx2).

This scheme also leads to a matrix system with entries

Ai,i1=Fθ,Ai,i=1+2Fθ,Ai,i+1=Fθ,

while right-hand side entry bi is

bi=uni+F(1θ)uni+12uni+uni1Δx2

The corresponding entries for the boundary points are as in the Backward Euler and Crank-Nicolson schemes listed earlier.

The Laplace and Poisson equation

The Laplace equation, 2u=0, or the Poisson equation, 2u=f, occur in numerous applications throughout science and engineering. In 1D these equations read u(x)=0 and u(x)=f(x), respectively. We can solve 1D variants of the Laplace equations with the listed software, because we can interpret uxx=0 as the limiting solution of ut=αuxx when u reach a steady state limit where ut0. Similarly, Poisson’s equation uxx=f arises from solving ut=uxx+f and letting t so ut0.

Technically in a program, we can simulate t by just taking one large time step, or equivalently, set α to a large value. All we need is to have F large. As F, we can from the schemes see that the limiting discrete equation becomes

un+1i+12un+1i+un+1i1Δx2=0,

which is nothing but the discretization [DxDxu]n+1i=0 of uxx=0.

The Backward Euler scheme can solve the limit equation directly and hence produce a solution of the 1D Laplace equation. With the Forward Euler scheme we must do the time stepping since F>1/2 is illegal and leads to instability. We may interpret this time stepping as solving the equation system from uxx by iterating on a time pseudo time variable.

Extensions

These extensions are performed exactly as for a wave equation as they only affect the spatial derivatives (which are the same as in the wave equation).

  • Variable coefficients
  • Neumann and Robin conditions
  • 2D and 3D

Future versions of this document will for completeness and independence of the wave equation document feature info on the three points. The Robin condition is new, but straightforward to handle:

αun=hT(uUs),[αDxu=hT(uUs)]ni

Analysis of schemes for the diffusion equation

Properties of the solution

A particular characteristic of diffusive processes, governed by an equation like

ut=αuxx,

is that the initial shape u(x,0)=I(x) spreads out in space with time, along with a decaying amplitude. Three different examples will illustrate the spreading of u in space and the decay in time.

Similarity solution

The diffusion equation (42) admits solutions that depend on η=(xc)/4αt for a given value of c. One particular solution is

u(x,t)=aerf(η)+b,

where

erf(η)=2πη0eζ2dζ,

is the error function, and a and b are arbitrary constants. The error function lies in (1,1), is odd around η=0, and goes relatively quickly to ±1:

limηerf(η)=1,limηerf(η)=1,erf(η)=erf(η),erf(0)=0,erf(2)=0.99532227,erf(3)=0.99997791.

As t0, the error function approaches a step function centered at x=c. For a diffusion problem posed on the unit interval [0,1], we may choose the step at x=1/2 (meaning c=1/2), a=1/2, b=1/2. Then

u(x,t)=12(1erf(x124αt))=12erfc(x124αt),

where we have introduced the complementary error function erfc(η)=1erf(η). The solution (45) implies the boundary conditions

u(0,t)=12(1erf(1/24αt)),
u(1,t)=12(1erf(1/24αt)).

For small enough t, u(0,t)1 and u(1,t)1, but as t, u(x,t)1/2 on [0,1].

Solution for a Gaussian pulse

The standard diffusion equation ut=αuxx admits a Gaussian function as solution:

u(x,t)=14παtexp((xc)24αt).

At t=0 this is a Dirac delta function, so for computational purposes one must start to view the solution at some time t=tϵ>0. Replacing t by tϵ+t in (48) makes it easy to operate with a (new) t that starts at t=0 with an initial condition with a finite width. The important feature of (48) is that the standard deviation σ of a sharp initial Gaussian pulse increases in time according to σ=2αt, making the pulse diffuse and flatten out.

Solution for a sine component

For example, (42) admits a solution of the form

u(x,t)=Qeatsin(kx).

The parameters Q and k can be freely chosen, while inserting (49) in (42) gives the constraint

a=αk2.

A very important feature is that the initial shape I(x)=Qsinkx undergoes a damping exp(αk2t), meaning that rapid oscillations in space, corresponding to large k, are very much faster dampened than slow oscillations in space, corresponding to small k. This feature leads to a smoothing of the initial condition with time.

The following examples illustrates the damping properties of (49). We consider the specific problem

ut=uxx,x(0,1), t(0,T],u(0,t)=u(1,t)=0,t(0,T],u(x,0)=sin(πx)+0.1sin(100πx).

The initial condition has been chosen such that adding two solutions like (49) constructs an analytical solution to the problem:

u(x,t)=eπ2tsin(πx)+0.1eπ2104tsin(100πx).

Figure Evolution of the solution of a diffusion problem: initial condition (upper left), 1/100 reduction of the small waves (upper right), 1/10 reduction of the long wave (lower left), and 1/100 reduction of the long wave (lower right) illustrates the rapid damping of rapid oscillations sin(100πx) and the very much slower damping of the slowly varying sin(πx) term. After about t=0.5104 the rapid oscillations do not have a visible amplitude, while we have to wait until t0.5 before the amplitude of the long wave sin(πx) becomes very small.

_images/diffusion_damping.png

Evolution of the solution of a diffusion problem: initial condition (upper left), 1/100 reduction of the small waves (upper right), 1/10 reduction of the long wave (lower left), and 1/100 reduction of the long wave (lower right)

Example: Diffusion of a discontinues profile

We shall see how different schemes predict the evolution of a discontinuous initial condition:

u(x,0)={UL,x<L/2UR,xL/2

Such a discontinuous initial condition may arise when two insulated blocks of metals at different temperature are brought in contact at t=0. Alternatively, signaling in the brain is based on release of a huge ion concentration on one side of a synapse, which implies diffusive transport of a discontinuous concentration function.

More to be written...

Analysis of discrete equations

A counterpart to (49) is the complex representation of the same function:

u(x,t)=Qeateikx,

where i=1 is the imaginary unit. We can add such functions, often referred to as wave components, to make a Fourier representation of a general solution of the diffusion equation:

u(x,t)kKbkeαk2teikx,

where K is a set of an infinite number of k values needed to construct the solution. In practice, however, the series is truncated and K is a finite set of k values need build a good approximate solution. Note that (50) is a special case of (51) where K={π,100π}, bπ=1, and b100π=0.1.

The amplitudes bk of the individual Fourier waves must be determined from the initial condition. At t=0 we have ukbkexp(ikx) and find K and bk such that

I(x)kKbkeikx.

(The relevant formulas for bk come from Fourier analysis, or equivalently, a least-squares method for approximating I(x) in a function space with basis exp(ikx).)

Much insight about the behavior of numerical methods can be obtained by investigating how a wave component exp(αk2t)exp(ikx) is treated by the numerical scheme. It appears that such wave components are also solutions of the schemes, but the damping factor exp(αk2t) varies among the schemes. To ease the forthcoming algebra, we write the damping factor as An. The exact amplification factor corresponding to A is Ae=exp(αk2Δt).

Analysis of the finite difference schemes

We have seen that a general solution of the diffusion equation can be built as a linear combination of basic components

eαk2teikx.

A fundamental question is whether such components are also solutions of the finite difference schemes. This is indeed the case, but the amplitude exp(αk2t) might be modified (which also happens when solving the ODE counterpart u=αu). We therefore look for numerical solutions of the form

unq=AneikqΔx=Aneikx,

where the amplification factor A must be determined by inserting the component into an actual scheme.

Stability

The exact amplification factor is Ae=exp(α2k2Δt). We should therefore require |A|<1 to have a decaying numerical solution as well. If 1A<0, An will change sign from time level to time level, and we get stable, non-physical oscillations in the numerical solutions that are not present in the exact solution.

Accuracy

To determine how accurately a finite difference scheme treats one wave component (53), we see that the basic deviation from the exact solution is reflected in how well An approximates Aen, or how well A approximates Ae. We can plot Ae and the various expressions for A, and we can make Taylor expansions of A/Ae to see the error more analytically.

Analysis of the Forward Euler scheme

The Forward Euler finite difference scheme for ut=αuxx can be written as

[D+tu=αDxDxu]nq.

Inserting a wave component (53) in the scheme demands calculating the terms

eikqΔx[D+tA]n=eikqΔxAnA1Δt,

and

AnDxDx[eikx]q=An(eikqΔx4Δx2sin2(kΔx2)).

Inserting these terms in the discrete equation and dividing by AneikqΔx leads to

A1Δt=α4Δx2sin2(kΔx2),

and consequently

A=14Fsin2(kΔx2),

where

F=αΔtΔx2

is the numerical Fourier number. The complete numerical solution is then

unq=(14Fsin2(kΔx2))neikqΔx.

Stability

We easily see that A1. However, the A can be less than 1, which will lead to growth of a numerical wave component. The criterion A1 implies

4Fsin2(p/2)2.

The worst case is when sin2(p/2)=1, so a sufficient criterion for stability is

F12,

or expressed as a condition on Δt:

ΔtΔx22α.

Note that halving the spatial mesh size, Δx12Δx, requires Δt to be reduced by a factor of 1/4. The method hence becomes very expensive for fine spatial meshes.

Accuracy

Since A is expressed in terms of F and the parameter we now call p=kΔx/2, we should also express Ae by F and p. The exponent in Ae is αk2Δt, which equals Fk2Δx2=F4p2. Consequently,

Ae=exp(αk2Δt)=exp(4Fp2).

All our A expressions as well as Ae are now functions of the two dimensionless parameters F and p.

Computing the Taylor series expansion of A/Ae in terms of F can easily be done with aid of sympy:

def A_exact(F, p):
    return exp(-4*F*p**2)

def A_FE(F, p):
    return 1 - 4*F*sin(p)**2

from sympy import *
F, p = symbols('F p')
A_err_FE = A_FE(F, p)/A_exact(F, p)
print A_err_FE.series(F, 0, 6)

The result is

AAe=14Fsin2p+2Fp216F2p2sin2p+8F2p4+

Recalling that F=αΔt/Δx, p=kΔx/2, and that sin2p1, we realize that the dominating error terms are at most

14αΔtΔx2+αΔt4α2Δt2+α2Δt2Δx2+.

Analysis of the Backward Euler scheme

Discretizing ut=αuxx by a Backward Euler scheme,

[Dtu=αDxDxu]nq,

and inserting a wave component (53), leads to calculations similar to those arising from the Forward Euler scheme, but since

eikqΔx[DtA]n=AneikqΔx1A1Δt,

we get

1A1Δt=α4Δx2sin2(kΔx2),

and then

A=(1+4Fsin2p)1.

The complete numerical solution can be written

unq=(1+4Fsin2p)neikqΔx.

Stability

We see from (59) that 0<A<1, which means that all numerical wave components are stable and non-oscillatory for any Δt>0.

Analysis of the Crank-Nicolson scheme

The Crank-Nicolson scheme can be written as

[Dtu=αDxDx¯ux]n+12q,

or

[Dtu]n+12q=12α([DxDxu]nq+[DxDxu]n+1q).

Inserting (53) in the time derivative approximation leads to

[DtAneikqΔx]n+12=An+12eikqΔxA12A12Δt=AneikqΔxA1Δt.

Inserting (53) in the other terms and dividing by AneikqΔx gives the relation

A1Δt=12α4Δx2sin2(kΔx2)(1+A),

and after some more algebra,

A=12Fsin2p1+2Fsin2p.

The exact numerical solution is hence

unq=(12Fsin2p1+2Fsin2p)neikpΔx.

Stability

The criteria A>1 and A<1 are fulfilled for any Δt>0.

Summary of accuracy of amplification factors

We can plot the various amplification factors against p=kΔx/2 for different choices of the F parameter. Figures Amplification factors for large time steps, Amplification factors for time steps around the Forward Euler stability limit, and Amplification factors for small time steps show how long and small waves are damped by the various schemes compared to the exact damping. As long as all schemes are stable, the amplification factor is positive, except for Crank-Nicolson when F>0.5.

_images/diffusion_A_F20_F2.png

Amplification factors for large time steps

_images/diffusion_A_F05_F025.png

Amplification factors for time steps around the Forward Euler stability limit

_images/diffusion_A_F01_F001.png

Amplification factors for small time steps

The effect of negative amplification factors is that An changes sign from one time level to the next, thereby giving rise to oscillations in time in an animation of the solution. We see from Figure Amplification factors for large time steps that for F=20, waves with pπ/2 undergo a damping close to 1, which means that the amplitude does not decay and that the wave component jumps up and down in time. For F=2 we have a damping of a factor of 0.5 from one time level to the next, which is very much smaller than the exact damping. Short waves will therefore fail to be effectively dampened. These waves will manifest themselves as high frequency oscillatory noise in the solution.

A value p=π/4 corresponds to four mesh points per wave length of eikx, while p=π/2 implies only two points per wave length, which is the smallest number of points we can have to represent the wave on the mesh.

To demonstrate the oscillatory behavior of the Crank-Nicolson scheme, we choose an initial condition that leads to short waves with significant amplitude. A discontinuous I(x) will in particular serve this purpose.

Run F=......

Exercise 1: Explore symmetry in a 1D problem

This exercise simulates the exact solution (48). Suppose for simplicity that c=0.

a) Formulate an initial-boundary value problem that has (48) as solution in the domain [L,L]. Use the exact solution (48) as Dirichlet condition at the boundaries. Simulate the diffusion of the Gaussian peak. Observe that the solution is symmetric around x=0.

b) Show from (48) that ux(c,t)=0. Since the solution is symmetric around x=c=0, we can solve the numerical problem in half of the domain, using a symmetry boundary condition ux=0 at x=0. Set up the initial-boundary value problem in this case. Simulate the diffusion problem in [0,L] and compare with the solution in a).

Filename: diffu_symmetric_gaussian.

Exercise 2: Investigate approximation errors from a ux=0 boundary condition

We consider the problem solved in Exercise 1: Explore symmetry in a 1D problem part b). The boundary condition ux(0,t)=0 can be implemented in two ways: 1) by a standard symmetric finite difference [D2xu]ni=0, or 2) by a one-sided difference [D+u=0]ni=0. Investigate the effect of these two conditions on the convergence rate in space.

Hint. If you use a Forward Euler scheme, choose a discretization parameter h=Δt=Δx2 and assume the error goes like Ehr. The error in the scheme is O(Δt,Δx2) so one should expect that the estimated r approaches 1. The question is if a one-sided difference approximation to ux(0,t)=0 destroys this convergence rate.

Filename: diffu_onesided_fd.

Exercise 3: Experiment with open boundary conditions in 1D

We address diffusion of a Gaussian function as in Exercise 1: Explore symmetry in a 1D problem, in the domain [0,L], but now we shall explore different types of boundary conditions on x=L. In real-life problems we do not know the exact solution on x=L and must use something simpler.

a) Imagine that we want to solve the problem numerically on [0,L], with a symmetry boundary condition ux=0 at x=0, but we do not know the exact solution and cannot of that reason assign a correct Dirichlet condition at x=L. One idea is to simply set u(L,t)=0 since this will be an accurate approximation before the diffused pulse reaches x=L and even thereafter it might be a satisfactory condition if the exact u has a small value. Let ue be the exact solution and let u be the solution of ut=αuxx with an initial Gaussian pulse and the boundary conditions ux(0,t)=u(L,t)=0. Derive a diffusion problem for the error e=ueu. Solve this problem numerically using an exact Dirichlet condition at x=L. Animate the evolution of the error and make a curve plot of the error measure

E(t)=L0e2dxL0udx.

Is this a suitable error measure for the present problem?

b) Instead of using u(L,t)=0 as approximate boundary condition for letting the diffused Gaussian pulse move out of our finite domain, one may try ux(L,t)=0 since the solution for large t is quite flat. Argue that this condition gives a completely wrong asymptotic solution as t0. To do this, integrate the diffusion equation from 0 to L, integrate uxx by parts (or use Gauss’ divergence theorem in 1D) to arrive at the important property

ddtL0u(x,t)dx=0,

implying that L0udx must be constant in time, and therefore

L0u(x,t)dx=L0I(x)dx.

The integral of the initial pulse is 1.

c) Another idea for an artificial boundary condition at x=L is to use a cooling law

αux=q(uuS),

where q is an unknown heat transfer coefficient and uS is the surrounding temperature in the medium outside of [0,L]. (Note that arguing that uS is approximately u(L,t) gives the ux=0 condition from the previous subexercise that is qualitatively wrong for large t.) Develop a diffusion problem for the error in the solution using (63) as boundary condition. Assume one can take uS=0 “outside the domain” since ue0 as x. Find a function q=q(t) such that the exact solution obeys the condition (63). Test some constant values of q and animate how the corresponding error function behaves. Also compute E(t) curves as defined above.

Filename: diffu_open_BC.

Exercise 4: Simulate a diffused Gaussian peak in 2D/3D

a) Generalize (48) to multi dimensions by assuming that one-dimensional solutions can be multiplied to solve ut=α2u. Set c=0 such that the peak of the Gaussian is at the origin.

b) One can from the exact solution show that ux=0 on x=0, uy=0 on y=0, and uz=0 on z=0. The approximately correct condition u=0 can be set on the remaining boundaries (say x=L, y=L, z=L), cf. Exercise 3: Experiment with open boundary conditions in 1D. Simulate a 2D case and make an animation of the diffused Gaussian peak.

c) The formulation in b) makes use of symmetry of the solution such that we can solve the problem in the first quadrant (2D) or octant (3D) only. To check that the symmetry assumption is correct, formulate the problem without symmetry in a domain [L,L]×[L,L] in 2D. Use u=0 as approximately correct boundary condition. Simulate the same case as in b), but in a four times as large domain. Make an animation and compare it with the one in b).

Filename: diffu_symmetric_gaussian_2D.

Exercise 5: Examine stability of a diffusion model with a source term

Consider a diffusion equation with a linear u term:

ut=αuxx+βu.

a) Derive in detail a Forward Euler scheme, a Backward Euler scheme, and a Crank-Nicolson for this type of diffusion model. Thereafter, formulate a θ-rule to summarize the three schemes.

b) Assume a solution like (49) and find the relation between a, k, α, and β.

Hint. Insert (49) in the PDE problem.

c) Calculate the stability of the Forward Euler scheme. Design numerical experiments to confirm the results.

Hint. Insert the discrete counterpart to (49) in the numerical scheme. Run experiments at the stability limit and slightly above.

d) Repeat c) for the Backward Euler scheme.

e) Repeat c) for the Crank-Nicolson scheme.

f) How does the extra term bu impact the accuracy of the three schemes?

Hint. For analysis of the accuracy, compare the numerical and exact amplification factors, in graphs and/or by Taylor series expansion.

Filename: diffu_stability_uterm.

Diffusion in heterogeneous media

Diffusion in heterogeneous media will normally imply a non-constant diffusion coefficient α=α(x). A 1D diffusion model with such a variable diffusion coefficient reads

ut=x(α(x)ux2),x(0,L), t(0,T]
u(x,0)=I(x),x[0,L]
\begin{split}\tag{66}
u(0,t)  = U_0, \quad  t>0,\end{split}
\begin{split}\tag{67}
u(L,t)  = U_L, \quad  t>0{\thinspace .}\end{split}

A short form of the diffusion equation with variable coefficients is ut=(αux)x.

Stationary solution

As t, the solution of the above problem will approach a stationary limit where u/t=0. The governing equation is then

ddx(αdudx)=0,

with boundary conditions u(0)=U0 and u(L)=uL. It is possible to obtain an exact solution of (68) for any α. Integrating twice and applying the boundary conditions to determine the integration constants gives

u(x)=U0+(ULU0)x0(α(ξ))1dξL0(α(ξ))1dξ.

Piecewise constant medium

Consider a medium built of M layers. The boundaries between the layers are denoted by b0,,bM, where b0=0 and bM=L. If the material in each layer potentially differs from the others, but is otherwise constant, we can express α as a piecewise constant function according to

\begin{split}\tag{70}
\alpha (x) = \left\lbrace\begin{array}{ll}
    \alpha_0,& b_0 \leq x < b_1,\\
    \vdots &\\
    \alpha_i,& b_i \leq x < b_{i+1},\\
    \vdots &\\
    \alpha_0,& b_{M-1} \leq x \leq b_M.
    \end{array}\right.\end{split}

The exact solution (69) in case of such a piecewise constant α function is easy to derive. Assume that x is in the m-th layer: x[bm,bm+1]. In the integral x0(a(ξ))1dξ we must integrate through the first m1 layers and then add the contribution from the remaining part xbm into the m-th layer:

u(x)=U0+(ULU0)m1j=0(bj+1bj)/α(bj)+(xbm)/α(bm)M1j=0(bj+1bj)/α(bj)

Remark. It may sound strange to have a discontinuous α in a differential equation where one is to differentiate, but a discontinuous α is compensated by a discontinuous ux such that αux is continues and therefore can be differentiated as (αux)x.

Implementation

Programming with piecewise function definition quickly becomes cumbersome as the most naive approach is to test for which interval x lies, and then start evaluating a formula like (71). In Python, vectorized expressions may help to speed up the computations. The convenience classes PiecewiseConstant and IntegratedPiecewiseConstant were made to simplify programming with functions like (70) and expressions like (71). These utilities not only represent piecewise constant functions, but also smoothed versions of them where the discontinuities can be smoothed out in a controlled fashion. This is advantageous in many computational contexts (although seldom for pure finite difference computations of the solution u).

The PiecewiseConstant class is created by sending in the domain as a 2-tuple or 2-list and a data object describing the boundaries b0,,bM and the corresponding function values α0,,αM1. More precisely, data is a nested list, where data[i][0] holds bi and data[i][1] holds the corresponding value αi, for i=0,,M1. Given bi and αi in arrays b and a, it is easy to fill out the nested list data.

In our application, we want to represent α and 1/α as piecewise constant function, in addition to the u(x) function which involves the integrals of 1/α. A class creating the functions we need and a method for evaluating u, can take the form

class SerialLayers:
    """
    b: coordinates of boundaries of layers, b[0] is left boundary
    and b[-1] is right boundary of the domain [0,L].
    a: values of the functions in each layer (len(a) = len(b)-1).
    U_0: u(x) value at left boundary x=0=b[0].
    U_L: u(x) value at right boundary x=L=b[0].
    """

    def __init__(self, a, b, U_0, U_L, eps=0):
        self.a, self.b = np.asarray(a), np.asarray(b)
        self.eps = eps  # smoothing parameter for smoothed a
        self.U_0, self.U_L = U_0, U_L

        a_data = [[bi, ai] for bi, ai in zip(self.b, self.a)]
        domain = [b[0], b[-1]]
        self.a_func = PiecewiseConstant(domain, a_data, eps)

        # inv_a = 1/a is needed in formulas
        inv_a_data = [[bi, 1./ai] for bi, ai in zip(self.b, self.a)]
        self.inv_a_func = \
             PiecewiseConstant(domain, inv_a_data, eps)
        self.integral_of_inv_a_func = \
             IntegratedPiecewiseConstant(domain, inv_a_data, eps)
        # Denominator in the exact formula is constant
        self.inv_a_0L = self.integral_of_inv_a_func(b[-1])

    def __call__(self, x):
        solution = self.U_0 + (self.U_L-self.U_0)*\
                   self.integral_of_inv_a_func(x)/self.inv_a_0L
        return solution

A visualization method is also convenient to have. Below we plot u(x) along with α(x) (which works well as long as maxα(x) is of the same size as maxu=max(U0,UL)).

class SerialLayers:
    ...

    def plot(self):
        x, y_a = self.a_func.plot()
        x = np.asarray(x); y_a = np.asarray(y_a)
        y_u = self.u_exact(x)
        import matplotlib.pyplot as plt
        plt.figure()
        plt.plot(x, y_u, 'b')
        plt.hold('on')  # Matlab style
        plt.plot(x, y_a, 'r')
        ymin = -0.1
        ymax = 1.2*max(y_u.max(), y_a.max())
        plt.axis([x[0], x[-1], ymin, ymax])
        plt.legend(['solution $u$', 'coefficient $a$'], loc='upper left')
        if self.eps > 0:
            plt.title('Smoothing eps: %s' % self.eps)
        plt.savefig('tmp.pdf')
        plt.savefig('tmp.png')
        plt.show()

Figure Solution of the stationary diffusion equation corresponding to a piecewise constant diffusion coefficient shows the case where

b = [0, 0.25, 0.5, 1]   # material boundaries
a = [0.2, 0.4, 4]       # material values
U_0 = 0.5;  U_L = 5     # boundary conditions
_images/flow_in_layers_case1.png

Solution of the stationary diffusion equation corresponding to a piecewise constant diffusion coefficient

By adding the eps parameter to the constructor of the SerialLayers class, we can experiment with smoothed versions of α and see the (small) impact on u. Figure Solution of the stationary diffusion equation corresponding to a smoothed piecewise constant diffusion coefficient shows the result.

_images/flow_in_layers_case1_eps.png

Solution of the stationary diffusion equation corresponding to a smoothed piecewise constant diffusion coefficient

Diffusion equation in axi-symmetric geometries

Suppose we have a diffusion process taking care in a straight tube with radius R. We assume axi-symmetry such that u is just a function of r and t. A model problem is

ut=1rr(rα(r)ur)+f(t),r(0,R), t(0,T],
ur(0,t)=0,t(0,T],
u(R,t)=0,t(0,T],
u(r,0)=I(r),r[0,R].

The condition (73) is a necessary symmetry condition at r=0, while (74) could be any Dirichlet or Neumann condition (or Robin condition in case of cooling or heating).

The finite difference approximation at r=0 of the spatial derivative term is the only new challenge in this problem. Let us in case of constant α expand the derivative to

2ur2+1rur.

The last term faces a difficulty at r=0 since it becomes a 0/0 expression because of the symmetry condition. L’Hosptial’s rule can be used:

limr01rur=limr02ur2.

The PDE at r=0 therefore becomes

ut=2α2ur2+f(t).

For a variable coefficient α(r) the expanded derivative reads

α(r)2ur2+1r(α(r)+rα(r))ur.

We have that the limit of a product is

limr01r(α(r)+rα(r))ur=limr0(α(r)+rα(r)) limxc1rur.

The second limit becomes as above, so the PDE at r=0, assuming (α(0)+rα(0))0, looks like

ut=(2α+rα)2ur2+f(t).

The second-order derivative is discretized in the usual way. Consider first constant α:

2α2r2u(r0,tn)[2α2DrDru]n0=2αun12un0+un1Δr2.

The fictitious value un1 can be eliminated using the discrete symmetry condition

[D2ru=0]n0un1=un1,

which then gives the modified approximation to the second-order derivative of u in r at r=0:

4αun1un0Δr2.

With variable α we simply get

(2α+rα)2DrDru]n0=(2α(0)+rα(0))un12un0+un1Δr2.

The discretization of the second-order derivative in r at another internal mesh point is straightforward:

1rr(rαur)|t=tnr=ri[r1Dr(rαDru)]ni=1Δr2(ri+12αi+12(uni+1uni)ri12αi12(uniuni1)).

θ-rule in time...

Diffusion equation in spherically-symmetric geometries

Discretization in spherical coordinates

Let us now pose the problem from the section Diffusion equation in axi-symmetric geometries in spherical coordinates, where u only depends on the radial coordinate r and time t. That is, we have spherical symmetry. For simplicity we restrict the diffusion coefficient α to be a constant. The PDE reads

ut=αrγr(rγur)+f(t),

for r(0,R) and t(0,T]. The parameter γ is 2 for spherically-symmetric problems and 1 for axi-symmetric problems. The boundary and initial conditions have the same mathematical form as in (72)-(75).

Since the PDE in spherical coordinates has the same form as the PDE in the section Diffusion equation in axi-symmetric geometries, just with the γ parameter being different, we can use the same discretization approach. At the origin r=0 we get problems with the term

γrut,

but L’Hosptial’s rule shows that this term equals γ2u/r2, and the PDE at r=0 becomes

ut=(γ+1)α2ur2+f(t).

Same discretization, write up with γ.

Discretization in Cartesian coordinates

The spherically-symmetric spatial derivative can be transformed to the Cartesian counterpart by introducing

v(r,t)=ru(r,t).

Inserting u=v/r in the PDE yields

1r2r(α(r)r2ut),

and then

r(dc2drvr+α2vr2)dc2drv.

The two terms in the parenthesis can be combined to

rr(αvr),

which is recognized as the variable-coefficient Laplace operator in one Cartesian coordinate.

Exercises

Exercise 6: Stabilizing the Crank-Nicolson method by Rannacher time stepping

It is well known that the Crank-Nicolson method may give rise to non-physical oscillations in the solution of diffusion equations if the initial data exhibit jumps (see the section Analysis of the Crank-Nicolson scheme). Rannacher [Ref1] suggested a stabilizing technique consisting of using the Backward Euler scheme for the first two time steps with step length 12Δt. One can generalize this idea to taking 2m time steps of size 12Δt with the Backward Euler method and then continuing with the Crank-Nicolson method, which is of second-order in time. The idea is that the high frequencies of the initial solution are quickly damped out, and the Backward Euler scheme treats these high frequencies correctly. Thereafter, the high frequency content of the solution is gone and the Crank-Nicolson method will do well.

Test this idea for m=1,2,3 on a diffusion problem with a discontinuous initial condition. Measure the convergence rate using the solution (45) with the boundary conditions (46)-(47) for t values such that the conditions are in the vicinity of ±1. For example, t<5a1.6102 makes the solution diffusion from a step to almost a straight line. The program diffu_erf_sol.py shows how to compute the analytical solution.

Project 7: Energy estimates for diffusion problems

This project concerns so-called energy estimates for diffusion problems that can be used for qualitative analytical insight and for verification of implementations.

a) We start with a 1D homogeneous diffusion equation with zero Dirichlet conditions:

ut=αuxx,xΩ=(0,L), t(0,T],
u(0,t)=u(L,t)=0,t(0,T],
u(x,0)=I(x),x[0,L].

The energy estimate for this problem reads

||u||L2||I||L2,

where the ||||L2 norm is defined by

||g||L2=L0g2dx.

The quantify ||u||L2 or 12||u||L2 is known as the energy of the solution, although it is not the physical energy of the system. A mathematical tradition has introduced the notion energy in this context.

The estimate (84) says that the “size of $u$” never exceeds that of the initial condition, or more equivalently, that the area under the u curve decreases with time.

To show (84), multiply the PDE by u and integrate from 0 to L. Use that uut can be expressed as the time derivative of u2 and that uxxu can integrated by parts to form an integrand u2x. Show that the time derivative of ||u||2L2 must be less than or equal to zero. Integrate this expression and derive (84).

b) Now we address a slightly different problem,

ut=αuxx+f(x,t),xΩ=(0,L), t(0,T],
u(0,t)=u(L,t)=0,t(0,T],
u(x,0)=0,x[0,L].

The associated energy estimate is

||u||L2||f||L2.

(This result is more difficult to derive.)

Now consider the compound problem with an initial condition I(x) and a right-hand side f(x,t):

ut=αuxx+f(x,t),xΩ=(0,L), t(0,T],
u(0,t)=u(L,t)=0,t(0,T],
u(x,0)=I(x),x[0,L].

Show that if w1 fulfills (81)-(83) and w2 fulfills (86)-(88), then u=w1+w2 is the solution of (90)-(92). Using the triangle inequality for norms,

||a+b||||a||+||b||,

show that the energy estimate for (90)-(92) becomes

||u||L2||I||L2+||f||L2.

c) One application of (93) is to prove uniqueness of the solution. Suppose u1 and u2 both fulfill (90)-(92). Show that u=u1u2 then fulfills (90)-(92) with f=0 and I=0. Use (93) to deduce that the energy must be zero for all times and therefore that u1=u2, which proves that the solution is unique.

d) Generalize (93) to a 2D/3D diffusion equation ut=(αu) for xΩ.

Hint. Use integration by parts in multi dimensions:

Ωu(αu)dx=Ωαuudx+Ωuαun,

where \frac{\partial u}{\partial n} = \boldsymbol{n}\cdot\nabla u, \boldsymbol{n} being the outward unit normal to the boundary \partial\Omega of the domain \Omega.

e) Now we also consider the multi-dimensional PDE u_t = \nabla\cdot (\alpha \nabla u). Integrate both sides over \Omega and use Gauss’ divergence theorem, \int_\Omega \nabla\cdot\boldsymbol{q}{\, \mathrm{d}x} = \int_{\partial\Omega}\boldsymbol{q}\cdot\boldsymbol{n}{\, \mathrm{d}s} for a vector field \boldsymbol{q}. Show that if we have homogeneous Neumann conditions on the boundary, \partial u/\partial n=0, area under the u surface remains constant in time and

\tag{94} \int_{\Omega} u{\, \mathrm{d}x} = \int_{\Omega} I{\, \mathrm{d}x} {\thinspace .}

f) Establish a code in 1D, 2D, or 3D that can solve a diffusion equation with a source term f, initial condition I, and zero Dirichlet or Neumann conditions on the whole boundary.

We can use (93) and (94) as a partial verification of the code. Choose some functions f and I and check that (93) is obeyed at any time when zero Dirichlet conditions are used. Iterate over the same I functions and check that (94) is fulfilled when using zero Neumann conditions.

g) Make a list of some possible bugs in the code, such as indexing errors in arrays, failure to set the correct boundary conditions, evaluation of a term at a wrong time level, and similar. For each of the bugs, see if the verification tests from the previous subexercise pass or fail. This investigation shows how strong the energy estimates and the estimate (94) are for pointing out errors in the implementation.

Filename: diffu_energy.