何かあれば GitHub のリポジトリに issue を作るか ryukau@gmail.com までお気軽にどうぞ。
Update: 2025-01-07
前回のシミュレーション では変形した式が Explicit FDM (Explicit Finite Difference Method) の形だったので、パラメータの値によっては発散するという問題がありました。 Implicit FDM を使えば計算コストと引き換えに発散しなくなります。
Explicit FDM と Implicit FDM では空間方向の有限差分の時間ステップが変わります。
例として1次元の波動方程式を変形します。ここでは時間 \(t\) までの値は得られているものとします。今から計算するのは \(u(\,\_\,, t + dt)\) の値です。
\[ \frac{\partial^2 u(x,\,t)}{\partial t^2} = c^2 \frac{\partial^2 u(x,\,t)}{\partial x^2} \]
Explicit FDM の形です。
\[ \frac{u{\left (x,t + dt \right )} - 2\,u{\left (x,t \right )} + u{\left (x,t - dt \right )}}{dt^{2}} = c^2\,\frac{u{\left (- dx + x,t \right )} - 2\,u{\left (x,t \right )} + u{\left (dx + x,t \right )}}{dx^{2}} \]
Explicit FDM の形では空間方向の微分について \(u(\,\_\,, t)\) の形で展開します。 \(u(\,\_\,, t + dt)\) の形になる項は時間方向の微分から出てくる \(u(x,t + dt)\) だけです。従って得られた有限差分の式を \(u(x,t + dt)\) について解けば、そのまま計算できる形になります。
Implicit FDM の形です。
\[ \frac{u{\left (x,t + dt \right )} - 2\,u{\left (x,t \right )} + u{\left (x,t - dt \right )}}{dt^{2}} = c^2\,\frac{u{\left (- dx + x,t + dt \right )} - 2\,u{\left (x,t + dt \right )} + u{\left (dx + x,t + dt \right )}}{dx^{2}} \]
Implicit FDM の形では空間方向の微分について \(u(\,\_\,, t + dt)\) の形で展開します。その結果 \(u(\,\_\,, t + dt)\) の形になる項が複数出てきます。1つの式を整理しても計算できる形にはなりませんが、すべての \(x\) について式を立てれば連立方程式として解くことができます。
1次元の波動方程式を Implicit FDM の形に変形して連立方程式を立てます。1次元の波動方程式を再掲します。
\[ \frac{\partial^2 u(x,\,t)}{\partial t^2} = c^2 \frac{\partial^2 u(x,\,t)}{\partial x^2} \]
SymPyを使ってImplicitな有限差分に変形します。
from sympy import *
= symbols('c t u x dx dt')
c, t, u, x, dx, dt
= u(x, t + dt)
u_xt1 = u_xt1.diff(x, x).as_finite_difference(dx)
u_dx2
= u(x, t)
u_xt = u_xt.diff(t, t).as_finite_difference(dt)
u_dt2
= Eq(u_dt2, c**2 * u_dx2)
eq = solve(eq, u(x, t + dt))
ans
0])) latex(expand(ans[
出力された式です。
\[ \begin{aligned} u{\left (x,t \right )} =& \frac{c^{2} dt^{2}}{dx^{2}} u{\left (x,dt + t \right )} - \frac{c^{2} dt^{2}}{2 dx^{2}} u{\left (- dx + x,dt + t \right )} - \frac{c^{2} dt^{2}}{2 dx^{2}} u{\left (dx + x,dt + t \right )}\\ &+ \frac{1}{2} u{\left (x,- dt + t \right )} + \frac{1}{2} u{\left (x,dt + t \right )} \end{aligned} \]
整理します。
\[ \begin{gathered} \alpha u{\left (x - dx,t + dt \right )} + \beta u{\left (x,t + dt \right )} + \alpha u{\left (x + dx,t + dt \right )} = \space u{\left (x,t \right )} - \frac{1}{2} u{\left (x,t - dt \right )}\\ \alpha = \space -\frac{c^{2} dt^{2}}{2 \; dx^{2}}, \quad \beta = \frac{c^{2} dt^{2}}{dx^{2}} + \space \frac{1}{2} \end{gathered} \]
連立方程式を立てます。
\[ \mathbf{A} \mathbf{u}^{t+1} = \mathbf{u}^t - 0.5 \mathbf{u}^{t-1} \]
\(\mathbf{A}\) と \(\mathbf{u}^t\) です。
\[ \mathbf{A} = \begin{bmatrix} \beta & l & 0 & 0 & \cdots & 0 & 0\\ \alpha & \beta & \alpha & 0 & \cdots & 0 & 0\\ 0 & \alpha & \beta & \alpha & \cdots & 0 & 0\\ & \vdots & & & & \vdots &\\ 0 & 0 & 0 & 0 & \cdots & r & \beta \end{bmatrix} ,\quad \mathbf{u}^t = \begin{bmatrix} u(x_0, t)\\ u(x_1, t)\\ u(x_2, t)\\ \vdots\\ u(x_{n-1}, t) \end{bmatrix} \]
\(\mathbf{A}\) に含まれる \(l\) と \(r\) です。
\[ \begin{aligned} l =& \alpha \left (1 + L \right )\\ r =& \alpha \left (1 + R \right ) \end{aligned} \]
\(L\) と \(R\) の値は境界条件によって決まります。固定端のときは \(L = R = 0\) 、自由端のときは \(L = R = 1\) となります。
立てた連立方程式を \(\mathbf{u}^{t+1}\) について解くことでシミュレーションを1ステップ進めることができます。
コードを実行すると1次元の波のシミュレーションを行い、結果を
wave1d.png
に画像として書き出します。
連立方程式のソルバにnumpy.linalg.solve、画像の書き出しにImageioを使っています。
import numpy
import imageio
class Wave1D():
def __init__(self, length, c, dx, dt, attenuation):
"""
1次元の波のシミュレータ。
:param length: 波を表す1次元配列の長さ。
:param c: 波の速度[m/s]。
:param dx: 配列の要素間の距離[m]。
:param dt: シミュレーションの1ステップで進む時間[s]。
:param attenuation: 厳密でない波の減衰係数。 [0, 1] の範囲。
"""
# u(x, t) -> self.wave[t][x]
self.length = length
self.wave = numpy.zeros((3, self.length))
self.c = c
self.dx = dx
self.dt = dt
self.attenuation = attenuation
self.refreshConstants()
self.initBoundary()
self.initMatrix()
self.pick(0, 0)
def value(self):
"""
描画用に最新の波を返す。
"""
return self.wave[0]
def refreshConstants(self):
"""
シミュレーションで用いる定数を設定。
"""
= (self.c * self.dt / self.dx)**2
c2_dt2_dx2 self.alpha = -c2_dt2_dx2 / 2
self.beta = 0.5 + c2_dt2_dx2
def initBoundary(self):
"""
境界条件を指定。 0 で固定端。 1 で自由端。
"""
self.L = 0
self.R = 1
def initMatrix(self):
"""
Implicit finite difference method で解く必要のある方程式の設定。
ax = b の a。
"""
= numpy.zeros((self.length, self.length))
mat
0][0] = self.beta
mat[0][1] = self.alpha * (1 + self.L)
mat[
= self.length - 1
last for i in range(1, last):
- 1] = self.alpha
mat[i][i = self.beta
mat[i][i] + 1] = self.alpha
mat[i][i
- 1] = self.alpha * (1 + self.R)
mat[last][last = self.beta
mat[last][last]
self.a = mat
def step(self):
"""
シミュレーションの1ステップ。
"""
self.wave = numpy.roll(self.wave, 1, axis=0)
if self.pickY != 0:
self.wave[1][self.pickX] = self.pickY
self.wave[0] = self.attenuation * numpy.linalg.solve(
self.a,
self.wave[1] - 0.5 * self.wave[2],
)
def reset(self):
"""
波を 0 で埋めて初期状態に戻す。
"""
self.wave.fill(0)
def pick(self, x, y):
"""
x, y で指定した位置の波をつまむ。
:param x: 波をつまむ場所。[0, 1] の範囲。
:param y: 波をつまむ高さ。任意の実数。
"""
self.pickX = int((self.length - 1) * numpy.clip(x, 0.0, 1.0))
self.pickY = y
if __name__ == "__main__":
= 512
length = Wave1D(length, 64, 0.1, 0.01, 1)
wave1d
= []
result 0.5, 1)
wave1d.pick(for t in range(0, length):
wave1d.step()
result.append(wave1d.value())
"wave1d.png", numpy.array(result)) imageio.imwrite(
実行結果の画像です。横はあるステップでの波の状態、縦は時間で上から下に向かって進んでいます。
Implicit FDMとExplicit FDMによる波のシミュレーションを比較した動画です。
デモのコードは次のリンクの draw.py
と
wave.py
になります。
Implicit FDM のシミュレーションでは減衰係数が1のときでも時間の経過とともに波がなまっていきます。
Explicit FDM と Implicit FDM では波の伝達速度も変わっています。