Published on

Julia で Lotka-Volterra 方程式を数値的に解く

Authors

Lotka-Volterra 方程式

NOTE

Lotka-Volterra 方程式には, 被食者-捕食者系, 競争系, 相利共生系等のモデルが存在するが, 本記事では被食者-捕食者系のみを扱う.

Lotka-Volterra 方程式とは, 生物の捕食-被食関係による個体数の変動を表現する数理モデルの一種であり, 以下で表される:

dxdt=axbxydydt=cy+μbxy\begin{align*} &\dfrac{dx}{dt}=ax-bxy\\ &\dfrac{dy}{dt}=-cy+\mu bxy \end{align*}

ここで, xx は被食者密度, yy は捕食者密度, aa は被食者の自然増加率, bb は捕食者の摂食率, cc は捕食者の自然死亡率, μ\mu は捕食者が摂食エネルギーを繁殖に転換する効率を表す.

解法

本記事では, Julia の DifferentialEquations1 を用いて解く.

モジュール読み込み

まず, 必要なモジュールを読み込む.

using DifferentialEquations
using Plots
plot_font = "Computer Modern"
default(fontfamily=plot_font, linewidth=1, label=nothing, grid=true)
REPL
julia> using DifferentialEquations

julia> using Plots

julia> plot_font = "Computer Modern"
"Computer Modern"

julia> default(fontfamily=plot_font, linewidth=1, label=nothing, grid=true)

問題設定

次に, Lotka-Volterra 方程式を Julia で定式化する.

function lotka_volterra!(du, u, p, t)
    a, b, c, μ = p
    x, y = u
    du[1] = a * x - b * x * y
    du[2] = μ * x * y - c * y
end
REPL
julia> function lotka_volterra!(du, u, p, t)
           a, b, c, μ = p
           x, y = u
           du[1] = a * x - b * x * y
           du[2] = μ * x * y - c * y
       end
lotka_volterra! (generic function with 1 method)

パラメータ設定

次に, パラメータを設定する.

a = 1.1
b = 0.4
c = 0.4
μ = 0.1

p = (a, b, c, μ)
REPL
julia> a = 1.1
1.1

julia> b = 0.4
0.4

julia> c = 0.4
0.4

julia> μ = 0.1
0.1

julia>

julia> p = (a, b, c, μ)
(1.1, 0.4, 0.4, 0.1)

初期条件と解の範囲の設定

次に, 初期条件と解の範囲を設定する.

u0 = [10.0, 5.0]
tspan = (0.0, 50.0)
REPL
julia> u0 = [10.0, 5.0]
2-element Vector{Float64}:
 10.0
  5.0

julia> tspan = (0.0, 50.0)
(0.0, 50.0)

問題の定義と解の計算

prob = ODEProblem(lotka_volterra!, u0, tspan, p)
sol = solve(prob)
REPL
julia> prob = ODEProblem(lotka_volterra!, u0, tspan, p)
ODEProblem with uType Vector{Float64} and tType Float64. In-place: true
timespan: (0.0, 50.0)
u0: 2-element Vector{Float64}:
 10.0
  5.0

julia> sol = solve(prob)
retcode: Success
Interpolation: specialized 4th order "free" interpolation, specialized 2nd order "free" stiffness-aware interpolation
t: 56-element Vector{Float64}:
  0.0
  0.10551052180168205
  0.3026858596029679
  0.544216162029267
  0.8667001358219422
  1.290752093822806
  1.7003476232387265
  2.2799589614331133
  2.882391006071913
  3.543472793323572
  4.326489593554792
  5.302659637431677
  6.963443099602529
 38.19836931152591
 39.838659848395615
 41.16261629602774
 42.54163919084672
 43.520484417635416
 44.36703417737004
 45.12268565170904
 45.803326845977956
 46.6942006009366
 47.56185147161654
 48.716821800426615
 50.0
u: 56-element Vector{Vector{Float64}}:
 [10.0, 5.0]
 [9.035627005169454, 5.299839898200966]
 [7.2519783179682475, 5.750076609572348]
 [5.3322876509632415, 6.0725687751425825]
 [3.445570572857435, 6.136242934717593]
 [1.9876933916021085, 5.791725958718605]
 [1.260857484176088, 5.246249684158559]
 [0.7793800583895246, 4.4050613386031285]
 [0.5774381211452372, 3.6027620299304046]
 [0.5098173175276516, 2.865039146276961]
 [0.5500951340758927, 2.181735189174335]
 [0.7795716913377725, 1.572474529264101]
 [2.118736661257719, 1.004176672543559]
 [1.0061244645004324, 1.3371372224811882]
 [2.9857121817598737, 0.9292591888113896]
 [7.767258822631018, 1.0644948778605032]
 [13.712147771527462, 2.9448359380604394]
 [7.006151179114605, 5.8233785729636]
 [2.2798679109384095, 5.929732938997305]
 [1.0060307334488037, 4.916269057596703]
 [0.6374632520786015, 3.95149560119708]
 [0.5046844403917583, 2.906203763361891]
 [0.5490656370547375, 2.1478912561942285]
 [0.8601521201225086, 1.462986213750023]
 [1.884380037612957, 1.0323197430186637]

結果のプロット

計算結果をプロットする.

plot(sol, xlabel="Time", ylabel="Population", label=["Prey" "Predator"], dpi=300)
REPL
julia> plot(sol, xlabel="Time", ylabel="Population", label=["Prey" "Predator"], dpi=300)

lv01

図1: 計算結果

解の軌道

解の軌道をプロットする.

plot(sol, idxs=(1, 2), xlabel="Prey Population", ylabel="Predator Population", label=false, dpi=300)
REPL
julia> plot(sol, idxs=(1, 2), xlabel="Prey Population", ylabel="Predator Population", label=false, dpi=300)

lv02

図2: 解の軌道

図2より, Lotka-Volterra 方程式の解が周期解 (閉軌道) となることが分かる.

まとめ

今回は, Julia の DifferentialEquations を用いて Lotka-Volterra 方程式を解いたが, DifferentialEquations が提供するソルバを利用することで, 数値解法を実装する必要がなく便利であった.

Footnotes

  1. Julia の DE ソルバモジュール.