) T# V: k: T1 b8 O 一、内容梗概
: J4 B$ Q2 d. e" X5 x2 p “Ocean Modelling for Beginners”第一章主要是一些编程语言和软件的介绍还有一些序言之类的内容,在此就不多做解释了。“The Decay Problem”是书中第二章最开始引入的一个问题,借此向读者介绍数值计算的基本知识。
6 p/ i/ l0 Z7 B" a “The Decay Problem” 的中文怎么称呼都好啦,衰退问题、腐败问题,总之就是假设某种物质的浓度是 C0C_0 ,这种物质的浓度随着时间变化会按照一定的速率逐渐减少,例如酒精挥发等等,这样一个过程可以由下面一个方程来表示: * T- z8 H/ o8 G, K* G; x
dCdt=−κ⋅C\frac{dC}{dt} = - \kappa \cdot C \\
- l7 s8 P( I' x" m0 Q; K. V" f 这是一个一阶常微分方程,其中变量 CC 表示某种物质的浓度, tt 代表时间, κ\kappa (希腊字母:kappa)是一个常量,影响衰减的速率。有了这样一个式子,我们就可以求解出任意时刻 tt 这种物质的浓度 C(t)C(t) 了。这个方程解起来也比较简单,对等式两边同时积分就能得到解,也就是未知函数 C(t)C(t) : 1 Z. b8 U4 [4 ^ T: L
C(t)=c⋅e−κ⋅tC(t) = c \cdot e^{-\kappa \cdot t} \\
7 D: Y8 Z* s: r+ v { 然而,由上式可以看出这个解不是唯一的(注意e前面的c指的是任意常数不是浓度),要确定唯一的解,还要在一些自变量点上给出 C(t)C(t) 的值,也就是边界条件。这里我们可以设 t0t_0 时刻物质的浓度 C(t0)=C0C(t_{0}) = C_0 : 6 F" G8 ?, T+ H! N, A, G2 `
{dCdt=−κ⋅C,t≥t0C(t0)=C0\left\{ \begin{array}{l} \frac{dC}{dt}= -\kappa \cdot C, t \geq t_0 \\ C(t_{0})=C_0 \end{array} \right. \\
4 |) S6 h3 q, y: a, m 在合理假定下,我们就可以认为从 t0t_0 时刻对应的初始浓度 C0C_0 开始,上式中的常微分方程决定了未知函数 C(t)C(t) 在 t_0">t>t0t > t_0 时的变化情况,也就是说这个边界条件可以确定常微分方程的唯一解。相应地,C(t0)=C0C(t_{0}) = C_0叫做初始条件,像上式一样带初始条件的常微分方程问题被称为初值问题。
6 n, S/ o! C* r: Q/ Z9 h 那么讲了这么多,该如何利用计算机来求解这样的问题呢?这里就需要介绍另一个概念——有限差分(finite difference)了,形如 f(x+b)−f(x+a)f(x+b)-f(x+a) 的式子称为有限差分,将上式除以 b−ab-a ,我们就得到了差商,在数值计算中用有限差分来近似替代微分是一种非常重要的方法,有限差分通常分为三种前向差分(forward difference)、后向差分(backward difference)和中心差分(central difference)。前向差分的形式为 f(x+h)−f(x)f(x+h) - f(x) ,其中 hh 是步长,假设步长为 Δt\Delta t ,我们可以将上面的常微分方程改写成前向差分的形式: / X6 Q: Y8 b8 p+ l
dCdt=C(tn+Δt)−C(tn)Δt=−κ⋅C(tn)\frac{dC}{dt} = \frac{C(t_n + \Delta t) - C(t_n)}{\Delta t} = -\kappa \cdot C(t_n) \\ 5 @' |+ s# B; d! z6 ^
换一种表示方式,让 C(tn)C(t_n) 为 CnC_n 也就是 tnt_n 时刻的浓度为 CnC_n ,则 tn+1t_{n+1} 时刻的物质浓度为C(tn+Δt)C(t_n + \Delta t) 即 Cn+1C_{n+1} :
. p$ m% p* T4 q% {; e+ u Cn+1−CnΔt=−κCn\frac{C_{n+1} - C_n}{\Delta t} = -\kappa C_n \\ 9 @' i; Y; Z) M1 f
整理一下左右两边的项可以得到:
3 t& x4 i: X. Z: F- L2 O8 h3 \+ q- k! Y1 e Cn+1=Cn−Δt⋅κ⋅Cn=(1−Δt⋅κ)CnC_{n+1} = C_n - \Delta t \cdot \kappa \cdot C_n = (1-\Delta t \cdot \kappa) C_n \\ : @$ G- m9 G( J$ I1 X _
这样的有限差分格式叫做显式格式(explicit scheme),利用上式,我们就可以从t0t_0开始“步进式”的计算出后续时刻的函数近似值,在计算机程序中利用一个循环就可以完成。细心的同学可能注意到在这种方法下 Δt\Delta t 和 κ\kappa 的值是不能随便取的,试想一下如果 1">Δt⋅κ>1\Delta t \cdot \kappa > 1 那么 t0t_0 时刻之后的浓度就会变成负数,现实中根本不可能存在这种情况。因此我们的时间步长 Δt\Delta t 和 κ\kappa 需要满足下面一个条件: ( i0 k4 v! E0 e; g- W2 E1 \! \
9 \8 K3 y# a0 J* o5 g% o1 H1 |
该条件被称为数值稳定性条件(condition of numerical stability)。换句话说,只有当上式被满足时,我们的预测值,也就是通过有限差分发计算出的t时刻的物质浓度,是稳定的。可以看出这种方法下, Δt\Delta t 的最大值要受 κ\kappa 的限制。
; t6 W# Z6 M9 K0 N+ g 除了上述的显式格式以外,还有两种格式,分别是隐式格式(implicit scheme)和混合格式(hybrid scheme)。 * p9 u7 [1 W6 H2 [ D+ w$ j
隐式格式
* E+ t% j3 Y% i1 s, I7 M4 ` 这种格式是通过后向差分得来的,后向差分其实就是用 xx 和 x−hx - h 替代显式差分中的 x+hx + h 和 xx ,将之前的微分方程写成后向差分的形式: 5 A# t. Y( y+ d+ Y% g
Cn+1−CnΔt=−κCn+1\frac{C_{n+1} - C_n}{\Delta t} = -\kappa C_{n+1} \\
4 _; X" r, G$ z# a2 j- I/ v7 Z8 ] 和之前一样整理下等式左右两边,得到了这样一个结果: 5 P5 H7 s1 N! G+ e# e% M( d
Cn+1=Cn(1+Δt⋅κ)C_{n+1} = \frac{C_n}{(1 + \Delta t \cdot \kappa)} \\ % ]6 Z, h' M! V1 \
与显式格式相比,一个很大的好处是 Δt\Delta t 的取值不再受到 κ\kappa 的限制了。而且分母始终是大于1的( 0" >κ>0\kappa >0 ),因此浓度一定会随着时间逐渐减少。
0 P. u4 A. i) } ? 混合格式
; Y0 N4 z" N7 ?( y% a* ?6 I 如字面意思,这种方法混合了上述两种格式,形式如下:
3 A8 K) b) s% z) ` Cn+1−CnΔt=−α⋅κ⋅Cn+1−(1−α)⋅κ⋅Cn\frac{C_{n+1} - C_n}{\Delta t} = - \alpha \cdot \kappa \cdot C_{n+1} - (1- \alpha) \cdot \kappa \cdot C_n \\ 0 k4 P, u* f# S
整理一下可以得到: 1 H$ H2 ]9 X- x1 f0 a; ~2 J
Cn+1=1−(1−α)⋅κ⋅Δt1+α⋅κ⋅ΔtCnC_{n+1} = \frac{1-(1-\alpha) \cdot \kappa \cdot \Delta t}{ 1 + \alpha \cdot \kappa \cdot \Delta t} C_n \\
: k2 @ X" m- i# m 其中 α\alpha (希腊字母:alpha)是一个权值,决定哪一种格式所占的比重大,当 alpha=1alpha =1 时这种方法完全变成了隐式格式,而当 α=0.5\alpha = 0.5 时,这种格式被称做半隐式格式(semi-implicit scheme)。 p, f3 [4 j% R4 j, v* Q
二、参考程序 G* Z1 k1 }9 y) V A$ d
上面就是“The Decay Problem”练习之前的大致内容和基本概念,接下来就进入编码阶段了。题目的要求是: κ=0.0001s−1\kappa = 0.0001 \quad s^{-1} , Δt=3600s\Delta t = 3600 s ,初始浓度 C0C_0 为100(%),模拟15个小时的浓度变化。 程序中我将三种方法都写了出来方便后面比较它们的异同,其中混合格式的 α\alpha 值取了0.5。同时程序中还包含了一个求 tt 时刻真实值的方法,利用的是文章一开始积分求出的 C(t)C(t) , c=C0c = C_0 。和猴子也能看懂的海洋数值模拟(零)里面说的一样,计算的部分用C++来写,数据处理和绘图利用python来完成,两部分代码如下:
7 L* f" b7 i8 Z) ^; M5 f. B* T- Z. N 数值模拟程序
* X2 A( k) N0 F/ v, Q2 _ #include <fstream>#include <cmath>#include <vector>#include <functional> p# d# r) H! g/ r
const double kappa = 0.0001f;/ Z0 U/ @% i+ x
const double alpha = 0.5f;
1 m# b5 K* b( _& e" T" m2 w$ c const double dt = 3600; // 3600 seconds( }) s4 K4 s0 h @4 D
: I6 A1 U# |' n( g: z9 P6 B5 b0 `/ I
// lambda expression+ S, ~0 v+ N) `0 y/ y j$ K# |/ }
auto explicit_scheme = [](double conc)->double{ return (1 - dt * kappa) * conc; };# b8 Y$ {3 s. @
auto implicit_scheme = [](double conc)->double{ return conc / (1 + dt * kappa); };3 ?5 t7 H1 u) C! d+ p4 E
auto hybrid_scheme = [](double conc)->double{ return (1 - (1 - alpha) * kappa * dt) * conc / (1 + alpha * kappa * dt); };
+ R; M* W9 v# Q% ~ |# v2 I0 \) N2 d8 f2 ]0 r' [9 C3 P
void Decay( double concentration,' ^; [- P; @9 \& w+ {0 F$ z Y' [
int steps,
( K2 H9 _2 G4 z: @ const std::function<double(double)>& scheme,' o5 k* ]( ]% t! W# `% y% R# {1 B
const char* result_path )
+ s4 e0 G% A2 J {
5 A7 j3 a: x) A' H5 s+ l3 O std::vector<double> res{concentration};
0 x. j5 X3 s9 b: k9 u4 A/ a for(int i = 0; i < steps; i++) ^5 w8 T, d* C& g2 t" Q( C
{
! D( @! R+ E; U9 K+ {" M double conc1 = scheme(res[i]);0 H) B) J, M+ F( t, t; n3 `. I
res.push_back(conc1);
5 Q8 g( H8 D# K# r }& O& u5 Z u! E7 p* X: V
: Y1 U! [9 U) g" v std::ofstream file(result_path);/ T) N2 \) W5 t; H
if(file.is_open())
* i6 ]1 J6 a% n, l" n8 q" x: j# U* { {
% x! Q% r3 z! ?# A; m. ^* y0 H for(auto val : res)+ h8 e- A/ H7 ^$ V
file << val << "\n";# [. J8 T; F8 {" M& S7 y" q
file.close();
2 V0 q! y) ^9 n3 W8 L! W }
' ]. j6 N: K" Z. j }
# W) p- q3 M: k) t" w) A6 {& V- e( B# G, o
void AnalyticalSolution(double concentration, int steps)
1 a" r* I! E% f* O& { {3 e# C. G6 B' c' G8 u( l
std::vector<double> res{concentration};" ?! W* Z9 U6 S
double t = 0;5 ?. S8 }& v1 O* b: |) N+ z
for(int i = 1; i <= steps; i++)
! J& W! n' {7 e+ o! ] {5 l' O, o, L }0 ?
t += dt;: q$ V/ a2 f3 j
double ct = std::exp(-kappa * t) * concentration;
9 a, o2 v5 U" \% |/ {# S res.push_back(ct);
6 F' U3 @' r* H& d( D; ]$ r }$ z" J3 l P* \/ R i/ i9 _
2 {9 L/ N9 z: E+ a7 S) W" v std::ofstream file("res_analytical.txt");% j9 X+ L, ^, d$ @6 r
if(file.is_open())4 Y/ u. S1 q/ G
{
3 ]( e3 d# t" D! G) O1 i- d8 z u for(auto val : res)
' P- c. `: F( s0 V file << val << "\n";+ h+ z9 M& L: Q5 `& ~, g: j
file.close();8 v6 q; w" v, Q, n) B
}' t( k3 D Q$ a4 t
}
1 k: E: l% F8 ?$ s* y4 L! S8 `, [8 W2 L5 {: p7 c s( t
int main()5 P' e7 C6 L# Y L% @. N4 @' y
{
( t; k9 U7 w/ R$ T& J( @ // the initial concentration is 100 %+ X. d: P9 k6 Y: ]" k/ O
double concentration = 100;
/ U0 v4 M% F2 L8 b* C' \ // which means the total simulating time is steps * dt seconds
' P$ y' a& W9 a7 V0 j9 ? int steps = 15;, i0 Y; Q5 Y: ^: e. h# J; d! o
Decay(concentration, steps, explicit_scheme, "res_explicit.txt");9 K* Y* |* l% d0 q4 `- F4 N( _! {
Decay(concentration, steps, implicit_scheme, "res_implicit.txt");
1 y+ g" i6 F8 h/ C% g Decay(concentration, steps, hybrid_scheme, "res_hybrid.txt");! x! b1 l) G8 u
AnalyticalSolution(concentration, steps);
' k- Z6 L- ~. |+ q9 k return 0;* W9 b+ C6 k4 ]2 }4 N0 Q. ~1 \4 K
}# Y! r! o) n$ u0 H4 B( s3 w# q
$ L$ B( ] B4 t' \$ W6 i: Y a
绘图程序
# W8 V. Z& d) k5 M5 ~1 `5 F import numpy as np1 }. t" @( N4 `4 k6 t1 f. }
import matplotlib.pyplot as plt4 M [& V6 t {9 ]( e# M% s
! n/ f+ D f% C5 h) S" L
plt.figure(Decay Problem)
- c8 H- e8 i' f" b( b, Y* J5 ]! |" `! \# w
res_explicit = np.loadtxt(res_explicit.txt).T8 _" \+ y# X% h" v5 R! s, _
res_implicit = np.loadtxt(res_implicit.txt).T% ?& O, n* B. M; u$ {& f
res_hybrid = np.loadtxt(res_hybrid.txt).T
$ m9 ?2 ~( N, F$ Y' {+ u( E analytical = np.loadtxt(res_analytical.txt).T, ~& i- W0 e) C0 A( P" D* ~
* ^; {% Y: V# N% S time = np.arange(0, 16, 1)( P# x: G7 Z& E$ h# @6 L3 X
plt.plot(time, analytical, color=black, linestyle=dashed, label=analytical): ~& p2 x9 H4 N3 A+ O7 c3 ^
plt.plot(time, res_explicit, label=explicit)
7 ?% \( n$ W2 s2 {7 Q) S0 w plt.plot(time, res_implicit, label=implicit)
$ u' b/ s( G( ~9 P plt.plot(time, res_hybrid, label=hybrid)
2 @ o4 Z1 _) V7 s# [$ p+ Y* `3 d! O: s. _* D2 L
plt.legend()8 g! T; J% G- F. z) q8 L
plt.title(Decay Problem)
. c6 i% q' z2 L! P) H plt.xlabel(Time (hours))
- C$ f% C' C u) V p4 ? plt.ylabel(Concentration (100%))
# y5 C1 ^: U1 B; s9 w3 F9 E plt.ylim(0, 100)7 ~ \% ?4 t; @9 ? ^2 e# R' |
plt.xlim(0, 15)1 ]2 m _4 j, F8 F# S' [! Q% S( O
plt.xticks(time)2 T! y* e; }, ~6 q) W
plt.grid(True)6 V5 V4 U4 Q6 b+ A7 V6 s
' ]- _7 \# k$ K% X& [ plt.show()
+ w: w& r4 b3 v% X, d' X, c
) E9 {9 f' Z" g3 c 三、运行结果
7 b; h% g9 ]) ~/ N9 q& X$ V) | 结果如下图1所示,程序模拟了初始浓度为100%的某物质从零时刻开始15个小时的浓度变化情况,分别采用了显式、隐式以及半隐式的差分方法,对比黑色虚线的真实值可以看出同一时刻显示差分法的计算出的浓度比真实值要少,相反,隐式差分法要多一些。令人惊讶的是半隐式差分法(hybrid),几乎和真实值的曲线吻合,从此图可以直观的看出三种不同的差分方法的区别。
/ Y% Y/ O$ w. V* _; b 图 1. Decay Problem程序模拟结果参考文献Jochen Kampf. Ocean Modelling for Beginners. Springer. 2009. 喻文健. 数值分析与算法(第2版). 清华大学出版社. 2015.Wikipedia. 有限差分. https://en.wikipedia.org/wiki/Finite_difference.之前的文章# L+ p6 u: ]" g- X1 _) S3 r
猴子也能看懂的海洋数值模拟(零) ( S/ Z5 j0 e) ?- D) W
& o. T; F p" B5 i$ p! M ~
: D8 N/ q( S# c8 D6 V
. I& E3 ~+ b a3 w$ |) H" ]9 k2 J2 t; u$ K
|