SolvingtheincompressibleNavierStokesequations PDF
SolvingtheincompressibleNavierStokesequations PDF
SolvingtheincompressibleNavierStokesequations PDF
1 Introduction
On the following pages you find a documentation for the Matlab program
mit18086 navierstokes.m, as it is used for Course 18.086: Computational Science and En-
gineering II at the Massachusetts Institute of Technology. The code can be downloaded from
the Computational Science and Engineering web page http://www-math.mit.edu/cse. It
is also linked on the course web page http://www-math.mit.edu/18086.
This code shall be used for teaching and learning about incompressible, viscous flows.
It is an example of a simple numerical method for solving the Navier-Stokes equations. It
contains fundamental components, such as discretization on a staggered grid, an implicit
viscosity step, a projection step, as well as the visualization of the solution over time. The
main priorities of the code are
1. Simplicity and compactness: The whole code is one single Matlab file of about
100 lines.
2. Flexibility: The code does not use spectral methods, thus can be modified to more
complex domains, boundary conditions, and flow laws.
3. Visualization: The evolution of the flow field is visualized while the simulation runs.
4. Computational speed: Full vectorization and pre-solving the arising linear systems
in an initialization step results in fast time stepping.
• variable box sizes, grid resolution, time step, and Reynolds numbers
1
• visualization of pressure field, velocity field and streamlines
• automatic transition from central to donor-cell discretization for the nonlinear advec-
tion part
• fast computation of the time-dependent solution for small to moderate mesh sizes
• 3d
• time-dependent geometries
• turbulence models
The code qualifies as a basis for modifications and extensions. The following extensions
have already been applied to the code by MIT students and other users:
• external forces
2
on a rectangular domain Ω = [0, lx ]×[0, ly ]. The four domain boundaries are denoted North,
South, West, and East. The domain is fixed in time, and we consider no-slip boundary
conditions on each wall, i.e.
A derivation of the Navier-Stokes equations can be found in [2]. The momentum equations
(1) and (2) describe the time evolution of the velocity field (u, v) under inertial and viscous
forces. The pressure p is a Lagrange multiplier to satisfy the incompressibility condition
(3). Note that the momentum equations are already put into a numerics-friendly form. The
nonlinear terms on the right hand side equal
which follows by the chain rule and equation (3). The above right hand side is often written
in vector form as (u · ∇)u. We choose to numerically discretize the form on the left hand
side, because it is closer to a conservation form.
The incompressibility condition is not a time evolution equation, but an algebraic condi-
tion. We incorporate this condition by using a projection approach [1]: Evolve the momen-
tum equations neglecting the pressure, then project onto the subspace of divergence-free
velocity fields.
3 Visualization
Every fixed number of time steps the current pressure and velocity field are visualized.
The pressure field is shown as a color plot with some contour lines. On top of this, the
normalized velocity field is shown as a quiver plot, i.e. little arrows indicate the direction of
the flow. Since in a lid driven cavity the flow rate varies significantly over different areas,
outputting the normalized velocity field is a common procedure. Additionally, stream lines
are shown. Those are paths particles would take in the flow if the velocity field was frozen
at the current instant in time. Since the velocity field is divergence-free, the streamlines are
closed. The stream lines are contour lines of the stream function q. It is a function whose
orthogonal gradient is the velocity field
3
4 Numerical Solution Approach
The general approach of the code is described in Section 6.7 in the book Computational
Science and Engineering [4].
While u, v, p and q are the solutions to the Navier-Stokes equations, we denote the
numerical approximations by capital letters. Assume we have the velocity field U n and V n
at the nth time step (time t), and condition (3) is satisfied. We find the solution at the
(n + 1)st time step (time t + ∆t) by the following three step approach:
1. Treat nonlinear terms
The nonlinear terms are treated explicitly. This circumvents the solution of a nonlinear
system, but introduces a CFL condition which limits the time step by a constant times
the spacial resolution.
U∗ − Un
= −((U n )2 )x − (U n V n )y (12)
∆t
V∗−Vn
= −(U n V n )x − ((V n )2 )y (13)
∆t
In Section 5 we will detail how to discretize the nonlinear terms.
2. Implicit viscosity
The viscosity terms are treated implicitly. If they were treated explicitly, we would
have a time step restriction proportional to the spacial discretization squared. We
have no such limitation for the implicit treatment. The price to pay is two linear
systems to be solved in each time step.
U ∗∗ − U ∗ 1
= (U ∗∗ + Uyy
∗∗
) (14)
∆t Re xx
V ∗∗ − V ∗ 1
= (V ∗∗ + Vyy
∗∗
) (15)
∆t Re xx
3. Pressure correction
We correct the intermediate velocity field (U ∗∗ , V ∗∗ ) by the gradient of a pressure
P n+1 to enforce incompressibility.
U n+1 − U ∗∗
= −(P n+1 )x (16)
∆t
V n+1 − V ∗∗
= −(P n+1 )y (17)
∆t
The pressure is denoted P n+1 , since it is only given implicitly. It is obtained by solving
a linear system. In vector notation the correction equations read as
1 n+1 1 n
U − U = −∇P n+1 (18)
∆t ∆t
Applying the divergence to both sides yields the linear system
1
−∆P n+1 = − ∇ · Un (19)
∆t
Hence, the pressure correction step is
4
(a) Compute F n = ∇ · Un
1
(b) Solve Poisson equation −∆P n+1 = − ∆t Fn
(c) Compute Gn+1 = ∇P n+1
(d) Update velocity field Un+1 = Un − ∆tGn+1
The question, which boundary conditions are appropriate for the Poisson equation
for the pressure P , is complicated. A standard approach is to prescribe homoge-
neous Neumann boundary conditions for P wherever no-slip boundary conditions are
prescribed for the velocity field. For the lid driven cavity problem this means that
homogeneous Neumann boundary conditions are prescribed everywhere. This implies
in particular that the pressure P is only defined up to a constant, which is fine, since
only the gradient of P enters the momentum equation.
In addition to the solution steps, we have the visualization step, in which the stream function
Qn is computed. Similarly to the pressure is is obtained by the following steps
1. Compute F n = (V n )x − (U n )y
5 Spacial Discretization
The spacial discretization is performed on a staggered grid with the pressure P in the cell
midpoints, the velocities U placed on the vertical cell interfaces, and the velocities V placed
on the horizontal cell interfaces. The stream function Q is defined on the cell corners.
Consider to have nx × ny cells. Figure 1 shows a staggered grid with nx = 5 and ny = 3.
When speaking of the fields P , U and V (and Q), care has to be taken about interior and
boundary points. Any point truly inside the domain is an interior point, while points on or
outside boundaries are boundary points. Dark markers in Figure 1 stand for interior points,
while light markers represent boundary points. The fields have the following sizes:
The values at boundary points are no unknown variables. For Dirichlet boundary conditions
they are prescribed, and for Neumann boundry conditions they can be expressed in term
of interior points. However, boundary points of U and V are used for the finite difference
approximation of the nonlinear advection terms. Note that the boundary points in the four
corner are never used.
5
U P
V
Here one or two of the neighboring points might be boundary points. The same
formula holds for the component V , and for the Laplacian of the pressure P and
the stream function Q. If the unknown quantity is stored in a large column vector,
then the above approximation can be represented as a large sparse block matrix being
applied from the left. Thus, solving the Poisson equations for P and Q, as well as
solving implicitly for the viscosity terms in U and V , yields sparse linear systems to
be solved, as detailed in Section 7.
• First derivatives
A first derivative in a grid point can be approximated by a centered stencil.
Ui+1,j − Ui−1,j
(Ux )i,j ≈ (21)
2hx
6
This, however, can yield instabilities, as shown in many textbooks on numerical anal-
ysis. Here the staggered grid comes into play. Assume, we are not interested in the
value of Ux in the position of Ui,j , but instead we want the value in the middle between
the points Ui+1,j and Ui,j . Then the approximation
Ui+1,j − Ui,j
(Ux )i+ 1 ,j ≈ (22)
2 hx
is a stable centered approximation to Ux in the middle between the two points. In
the staggered grid this position happens to be the position of Pi,j .
– Pressure correction
In the pressure correction, the approximation of first derivatives on a staggered
grid works out perfectly:
1. The divergence of the velocity field F = ∇ · U computes Ux and Vy . Both
values then live in the cell centers, i.e. they can be added directly.
1
2. For the pressure Poisson equation −∆P = − ∆t F , both sides are defined in
the cell centers, and −∆P can be approximated as described above.
3. The gradient of the pressure G = ∇P requires the computation of Px and
Py . Again, a look at Figure 1 indicates that the former then live at the
position of U , while the latter live at the positions of V . Exactly where both
are needed to update the velocity field.
– Stream function
Similarly, the staggered grid works fine for the stream function. Both Vx and Uy
live on the cell corners, so the Poisson right hand side F = Vx − Uy and thus the
stream function Q lives at the cell corners.
– Nonlinear terms (central differencing)
The nonlinear terms are the only place where the discretization on the staggered
grid does not work directly. For instance, the product U V is not directly defined,
since U and V live in different positions. The solution is to take the arguments
backwards: For updating U , we need (U 2 )x and (U V )y . If the flow in each time
step is comparably slow, we wish to use the same centered staggered derivatives
as before. This requires U 2 to be defined in the cell centers, and U V to the defined
in the cell corners. We obtain these quantities by interpolating two neighboring
values.
Ui,j + Ui+1,j 2
2
(U )i+ 1 ,j = (23)
2 2
Ui,j + Ui,j+1
Ui,j+ 1 = (24)
2 2
Vi,j + Vi+1,j
Vi+ 1 ,j = (25)
2 2
Analogously for the component V . The index notation lets this idea look more
complicated than it is. All we do is create new data between two points by av-
eraging. Using an overbar with superscript h to indicate a horizontally averaged
7
quantity, and an overbar with superscript v to indicate a vertically averaged
quantity, we can write the update for the nonlinear terms as
U∗ − U
= −((Ū h )2 )x − (Ū v V̄ h )y (26)
∆t
V∗−V
= −(Ū v V̄ h )x − ((V̄ v )2 )y (27)
∆t
– Nonlinear terms (upwinding)
The above centered differencing is appropriate if quantities are not transported
too far in each time step. For faster flows or larger time steps, the discretization
shall be closer to an upwinding approach. We implement a smooth transition
between centered differencing and upwinding using a parameter γ ∈ [0, 1]. We
define it as
γ = min 1.2 · ∆t · max max |Ui,j |, max |Vi,j | , 1 (28)
i,j i,j
The value of gamma is the maximum fraction of a cells which information can
travel in one time step, multiplied by 1.2, and capped by 1. The factor of 1.2 is
taken from the experience that often times tending a bit more towards upwinding
can be advantageous for accuracy [3].
The linear combination between centered differencing and upwinding is imple-
mented in the following way. In corresponence to the averaged quantities, we
define differenced quantities
Ui+1,j − Ui,j
(Ũ h )i+ 1 ,j = (29)
2 2
Ui,j+1 − Ui,j
(Ũ v )i,j+ 1 = (30)
2 2
and analogously for Ṽ h and Ṽ v . Using the linear combination, equations (26)
and (27) generalize to
U∗ − U
= − (Ū h )2 − γ|Ū h |Ũ h − (Ū v V̄ h ) − γ|V̄ h |Ũ v (31)
∆t x y
V∗−V
= − (Ū v V̄ h ) − γ|Ū v |Ṽ h − (V̄ v )2 − γ|V̄ v |Ṽ v (32)
∆t x y
One can check the approximation values, as for instance for the U 2 term:
(Ū h )2 − γ|Ū h |Ũ h = |Ū h
| |Ū h
| − γ Ũ h
(33)
i+ 12 ,j i+ 12 ,j
1−γ Ui+1,j + 1+γ Ui,j if Ū h ≥ 0
2 2
= Ū h 1+γ (34)
2 Ui+1,j + 1−γ 2 Ui,j if Ū h < 0
One can easily see that this becomes averaged central differencing for γ = 0 and
conservative upwinding for γ = 1.
8
5.2 Boundary conditions
In the lid driven cavity problem we provide Dirichlet boundary conditions for U and V , and
Neumann boundary conditions for P . Applying the correct boundary conditions requires
care, since for the staggered grid some points lie on a boundary while others have a boundary
between them. At the points that lie on the boundary the value is directly prescribed, such
as U at the west and east boundary, and V at the north and south boundary. For U at the
north and south boundary and for V at west and east boundary, one has to define a value
between two data points. The idea is the same as above: average the values of two points.
For instance, the north boundary lies between points with velocity U . Let the two points
below respectively above the boundary be Ui,j and Ui,j+1 , and let the prescribed boundary
value be UN . Then the boundary condition is
Ui,j + Ui,j+1
= UN ⇐⇒ Ui,j + Ui,j+1 = 2UN (35)
2
Analogously at the south boundary, and for V at the west and east boundary.
The normal derivative ∂P
∂n at the boundary is defined using two points which have the
boundary in their middle. For instance, at the north boundary, prescribing homogeneous
Neumann boundary conditions yields
Pi,j+1 − Pi,j
=0 ⇐⇒ Pi,j+1 = Pi,j (36)
hy
The stream function is defined on cell boundary points. Thus all its boundary points lie
on the domain boundary. We prescibe homogeneous Dirichlet boundary conditions, which
can be prescribed directly.
9
In the code the third approach is implemented. Since the matrices are symmetric positive
definite, the sparse Cholesky decomposition can be used.
7 Implementation in Matlab
We take advantage of the Matlab data structures and save the field quantities as matrices.
Each quantity is stored without boundary points, yielding matrices of the following sizes.
quantity matrix size
U (nx − 1) × (ny + 1)
V (nx + 1) × (ny − 1)
P nx × ny
Q (nx − 1) × (ny − 1)
In solving for P and Q, as well as for the viscosity part for U and V , the boundary conditions
are brought to the right hand side. The only time boundary values are used explicitly is
when evaluating the nonlinear terms. Here, we extend the matrices U and V to matrices
containing also the boundary points.
The boundary points which fall onto a boundary are concatenated directly. The
values of the boundary points half a meshsize outside of the boundary are computed
using relation (35). These extended matrices U e and V e are used to proceed further.
Compute (U V )x and (U V )y by averaging U vertically and V horizontally, and using
the transition parameter γ:
Ua = avg(Ue’)’; Ud = diff(Ue’)’/2;
Va = avg(Ve); Vd = diff(Ve)/2;
UVx = diff(Ua.*Va-gamma*abs(Ua).*Vd)/hx;
UVy = diff((Ua.*Va-gamma*Ud.*abs(Va))’)’/hy;
10
Compute (U 2 )x and (V 2 )y by averaging U horizontally and V vertically:
Ua = avg(Ue(:,2:end-1)); Ud = diff(Ue(:,2:end-1))/2;
Va = avg(Ve(2:end-1,:)’)’; Vd = diff(Ve(2:end-1,:)’)’/2;
U2x = diff(Ua.^2-gamma*abs(Ua).*Ud)/hx;
V2y = diff((Va.^2-gamma*abs(Va).*Vd)’)’/hy;
U = U-dt*(UVy(2:end-1,:)+U2x);
V = V-dt*(UVx(:,2:end-1)+V2y);
• Pressure correction
Compute divergence of the velocity field ∇ · U:
diff(U(:,2:end-1))/hx+diff(V(2:end-1,:)’)’/hy
U(2:end-1,2:end-1) = U(2:end-1,2:end-1)-diff(P)/hx;
V(2:end-1,2:end-1) = V(2:end-1,2:end-1)-diff(P’)’/hy;
• Stream function
Compute curl of the velocity field ∇ × U:
diff(U’)’/hy-diff(V)/hx
• Implicit viscosity
Consider interior points only. Add appropriate boundary values (for implicit viscosity
step). Reshape. Solve linear system. Reshape and update interior points:
rhs = reshape(U(2:end-1,2:end-1)+Ubc,[],1);
u(peru) = Ru\(Rut\rhs(peru));
U(2:end-1,2:end-1) = reshape(u,nx-1,ny);
• Pressure correction
Compute right hand side in matrix form. Reshape. Solve linear system. Reshape
back and update interior points:
11
rhs = reshape(diff(U(:,2:end-1))/hx+diff(V(2:end-1,:)’)’/hy,[],1);
p(perp) = -Rp\(Rpt\rhs(perp));
P = reshape(p,nx,ny);
U(2:end-1,2:end-1) = U(2:end-1,2:end-1)-diff(P)/hx;
V(2:end-1,2:end-1) = V(2:end-1,2:end-1)-diff(P’)’/hy;
• Stream function
Compute right hand side in matrix form. Reshape. Solve linear system. Reshape
back:
rhs = reshape(diff(U’)’/hy-diff(V)/hx,[],1);
q(perq) = Rq\(Rqt\rhs(perq));
Q = zeros(nx+1,ny+1);
Q(2:end-1,2:end-1) = reshape(q,nx-1,ny-1);
For visualization purposes, the homogeneous Dirichlet boundary points are included
in the matrix Q.
12
a11 type of boundary condition
1 Neumann
2 Dirichlet for point on the boundary
3 Dirichlet for boundary between two points
This leaves us with one small detail: We prescribe Neumann boundary conditions for P .
Hence, the matrix Lp is of corank 1. Its nullspace is N (Lp) = span(~e), where ~e = (1, . . . , 1)T .
Since Lp is symmetric, its range is R(Lp) = {~b : ~eT ~b = 0}. The right hand side is in the range
of Lp by construction of the projection step. In order to perform a Cholesky factorization,
we modify Lp so that it becomes a regular matrix, while still yielding a correct solution for
the pressure. This is achieved by adding 1 to the last entry. The proof is as follows.
~ = (0, . . . , 0, 1)T . Thus w
Let w ~w~ T is a matrix with its only nonzero entry being a 1 in
the last position. Let A be the Neumann-Poisson matrix, i.e. A~e = 0. The new modified
Poisson matrix is B = A + w ~w~ T . Let ~u be an arbitrary vector. Then
B~u = |{z} ~ T ~u) |{z}
A~u +(w w
~ . (38)
⊥~e 6⊥~e
This sum can only equal 0 if ~u = 0. Hence B has full rank. It remains to show that for any
right hand side ~b, which is admissible (i.e. ~b ⊥ ~e), the solution to B~u = ~b satisfies A~u = ~b.
First note that due to (38), we have B −1 w ~ = ~e. Hence, we get
AB −1 = (B − w
~w~ T )B −1 = I − w
~w~ T B −1 = I − w~
~ eT ,
and consequently AB −1~b = ~b.
13
8 Matlab Code
The code can be downloaded from http://www-math.mit.edu/18086 and
http://www-math.mit.edu/cse
function mit18086_navierstokes
%MIT18086_NAVIERSTOKES
% Solves the incompressible Navier-Stokes equations in a
% rectangular domain with prescribed velocities along the
% boundary. The solution method is finite differencing on
% a staggered grid with implicit diffusion and a Chorin
% projection method for the pressure.
% Visualization is done by a colormap-isoline plot for
% pressure and normalized quiver and streamline plot for
% the velocity field.
% The standard setup solves a lid driven cavity problem.
fprintf(’initialization’)
Lp = kron(speye(ny),K1(nx,hx,1))+kron(K1(ny,hy,1),speye(nx));
Lp(1,1) = 3/2*Lp(1,1);
perp = symamd(Lp); Rp = chol(Lp(perp,perp)); Rpt = Rp’;
Lu = speye((nx-1)*ny)+dt/Re*(kron(speye(ny),K1(nx-1,hx,2))+...
kron(K1(ny,hy,3),speye(nx-1)));
peru = symamd(Lu); Ru = chol(Lu(peru,peru)); Rut = Ru’;
14
Lv = speye(nx*(ny-1))+dt/Re*(kron(speye(ny-1),K1(nx,hx,3))+...
kron(K1(ny-1,hy,2),speye(nx)));
perv = symamd(Lv); Rv = chol(Lv(perv,perv)); Rvt = Rv’;
Lq = kron(speye(ny-1),K1(nx-1,hx,2))+kron(K1(ny-1,hy,2),speye(nx-1));
perq = symamd(Lq); Rq = chol(Lq(perq,perq)); Rqt = Rq’;
% implicit viscosity
rhs = reshape(U+Ubc,[],1);
u(peru) = Ru\(Rut\rhs(peru));
U = reshape(u,nx-1,ny);
rhs = reshape(V+Vbc,[],1);
v(perv) = Rv\(Rvt\rhs(perv));
V = reshape(v,nx,ny-1);
% pressure correction
rhs = reshape(diff([uW;U;uE])/hx+diff([vS’ V vN’]’)’/hy,[],1);
p(perp) = -Rp\(Rpt\rhs(perp));
P = reshape(p,nx,ny);
U = U-diff(P)/hx;
V = V-diff(P’)’/hy;
% visualization
if floor(25*k/nt)>floor(25*(k-1)/nt), fprintf(’.’), end
if k==1|floor(nsteps*k/nt)>floor(nsteps*(k-1)/nt)
% stream function
rhs = reshape(diff(U’)’/hy-diff(V)/hx,[],1);
q(perq) = Rq\(Rqt\rhs(perq));
Q = zeros(nx+1,ny+1);
Q(2:end-1,2:end-1) = reshape(q,nx-1,ny-1);
clf, contourf(avg(x),avg(y),P’,20,’w-’), hold on
contour(x,y,Q’,20,’k-’);
Ue = [uS’ avg([uW;U;uE]’)’ uN’];
Ve = [vW;avg([vS’ V vN’]);vE];
Len = sqrt(Ue.^2+Ve.^2+eps);
quiver(x,y,(Ue./Len)’,(Ve./Len)’,.4,’k-’)
hold off, axis equal, axis([0 lx 0 ly])
p = sort(p); caxis(p([8 end-7]))
15
title(sprintf(’Re = %0.1g t = %0.2g’,Re,k*dt))
drawnow
end
end
fprintf(’\n’)
%=======================================================================
function B = avg(A,k)
if nargin<2, k = 1; end
if size(A,1)==1, A = A’; end
if k<2, B = (A(2:end,:)+A(1:end-1,:))/2; else, B = avg(A,k-1); end
if size(A,2)==1, B = B’; end
function A = K1(n,h,a11)
% a11: Neumann=1, Dirichlet=2, Dirichlet mid=3;
A = spdiags([-1 a11 0;ones(n-2,1)*[-1 2 -1];0 a11 -1],-1:1,n,n)’/h^2;
Acknowledgments
The author is thankful to Per-Olof Persson and Jean-Christophe Nave for their useful com-
ments, which helped make the code more efficient, and to Gilbert Strang for his helpful
remarks on notation and presentation.
References
[1] A.J. Chorin, Numerical solution of the Navier-Stokes equations. Math. Comput. 22,
745–762, 1968.
[2] A.J. Chorin, J.E. Marsden, A mathematical introduction to fluid mechanics. Third
edition, Springer, 2000.
16