!!! サイト改修中のため表示が乱れる場合があります(1月末頃まで) !!!
ライフゲーム / 人工生命

17-11. セルオートマトンで氷の剣山「ペニテンテ」をシミュレーションしてみた

やること

ペニテンテ(Penitentes)は、アンデス山脈などで見られる、地面から雪や氷がニョキニョキ生えてるー!的な自然現象です。

『自然が魅せる氷の剣山“ペニテンテ”』
まるで針のムシロを彷彿させる 険しい氷の剣山が一面に広がる不思議な光景は ヒマラヤ山脈/アンデス山脈/アフリカ・キリマンジャロなど 低緯度かつ標高4,00…
Penitente (snow formation) - Wikipedia

地面から生えるのではなく雪が融けることによって形成されるようで、一旦くぼみができ始めると日光が入り込んで融解が促進される「正のフィードバック」が生じるためと言われています。

今回はペニテンテをセルオートマトンで簡略的にモデル化し、シミュレーションしてみます

実行環境

WinPython3.6をおすすめしています。

WinPython - Browse /WinPython_3.6/3.6.7.0 at SourceForge.net
Portable Scientific Python 2/3 32/64bit Distribution for Windows

Google Colaboratoryが利用可能です。

Google Colab

モデル

「雪(bool)」と「雪の耐久値(float)」をそれぞれ2次元配列で用意し、ライフゲームのように各セルを更新していきます。以下は実際のペニテンテの原理に忠実に則ったものではありませんので、簡略的なモデルとお考えください。

まず、雪を融かす要因として、雪マスの上側のU字型の空間に着目します。このような形状がある場合、壁から反射してきた日光によって融解が促進されると考えられるため、オレンジ線1本につき雪の耐久値が -0.1 されるとします。

次に、雪を強固にする要因として、雪マスの側面に着目します。このように側面が空いている場合、横殴りの吹雪によって雪が融けにくくなると考え、青線1本につき雪の耐久値が +0.1 されるとします。

あまり厳密に考えないで、アバウトに行きます。次のような雪の状態を考え、いくつか適当なセルについて上記の仮定に従って1ステップあたりの雪の耐久値の変化量を書き込みます。

耐久値は0~1の範囲で変動し、0になった雪セルは融けます。

あとは畳み込みのフィルターの各係数をうーんとひねり出します。眠くてひねり出せない場合は連立方程式を立てて解きましょう(モデルがアバウトなので連立方程式もアバウトに解くことが大切です)。こんな感じになりました。

中央と下の係数は一意に定まらず、合計が +0.1 になればいいようです。それではシミュレーションしてみましょう。

コード

おなじみ scipy.signal.convolve2d() を使ってサクッとセルを更新していきます。初期状態をランダムにし、またステップ毎に雪の耐久値にランダムな摂動を加えているので、実行の度に異なる結果になります。詳細はコメントをご参照ください。

import numpy as np
import numpy.random as nr
import matplotlib.pyplot as plt
from scipy import signal

#============================
# 設定
#============================

#フィールドサイズ
h, w = 50, 100


#============================
# 表示関数
#============================
def show():
    plt.figure(figsize=(10, 10))
    plt.imshow(s, cmap='Blues', vmin=0, vmax=1)
    plt.show(), print()


#============================
# メイン
#============================

#雪の初期化
s = np.ones((h, w), dtype=bool) #0:なし, 1:雪
s[0, :] = 0 #最上段は溶かしておく
s[1, :] = nr.randint(0, 2, w) #2段目はランダムにしておく

#雪の耐久値の初期化
t = np.ones((h, w)) #耐久値は1.0から開始

#畳み込み用フィルタ
f = np.array([[-0.1, +0.3, -0.1],
              [+0.1, -0.2, +0.1],
              [-0.2, +0.3, -0.2]])

#初期状態の表示
show()

#状態の更新
for step in range(1, 999):
    #雪×フィルターで畳み込んで耐久値を更新
    t = t + signal.convolve2d(s, f, mode='same', boundary='fill', fillvalue=1) #範囲外は雪ありとする
    
    #耐久値はランダムに減少する(ゆらぎ)
    t -= nr.rand(h, w) * 0.01
    
    #耐久値が尽きた雪は融ける
    s[t<0] = 0
    
    #耐久値は0~1に収める
    t[t<0] = 0
    t[t>1] = 1
    
    #表示
    show()
    
    #下から5行目が融け始めたら終了
    if np.sum(s[-6, :]) < w:
        break

結果

畳み込みフィルターの中央と下の係数を4パターン試してみました。

中央の係数:-0.2、下の係数:+0.3
中央の係数:-0.4、下の係数:+0.5
中央の係数:-0.6、下の係数:+0.7
中央の係数:-0.8、下の係数:+0.9

まとめ

結果の3番目くらいが私は好きですね。ごく簡単なモデルですが、南米アンデスのペニテンテを再現できたと思います。

リアクションのお願い

「参考になった!」「刺激された!」と思ったらぜひリアクションをしましょう。エンジニアの世界はGive and Takeによって成り立っています。これからも無料で良質な情報にアクセスできるよう、Giveする人への感謝をリアクションで示しましょう!

この記事をシェアする

自身のブログ等で使用する場合は引用を忘れずに!

また、寄付も受け付けています。コーヒー1杯でとても喜びます(*˘︶˘*)

 Amazonでギフト券(アマギフ)を贈る

こちらのリンク から金額を指定してお贈りください。(デフォルトで10000円になっているのでご変更ください)

配送:Eメール
受取人:staffあっとvigne-cla.com
贈り主:あなたのお名前やニックネーム
メッセージ:◯◯の記事が参考になりました。など

のようにご入力ください。見返りはありませんのでご了承ください。

 Amazonで食事券(すかいらーく優待券)を贈る

500円 1000円 2000円 5000円 からお贈りください。

配送:Eメール
受取人:staffあっとvigne-cla.com
贈り主:あなたのお名前やニックネーム
メッセージ:◯◯の記事が参考になりました。など

のようにご入力ください。見返りはありませんのでご了承ください。

 その他、ギフト券やクーポン券をメールで贈る

デジタルのギフト券/クーポン券はメールアドレス(staffあっとvigne-cla.com)までお送りください。受領の返信をいたします。
紙のギフト券/クーポン券は 「郵便物はこちらへ」の住所 まで送付してください。名刺やメールアドレスを同封していただければ受領の連絡をいたします。
余った株主優待券等の処理におすすめです。
いずれも見返りはありませんのでご了承ください。

不明点はSNSでお気軽にご連絡ください

ビネクラのTwitter・Youtubeでコメントをください!


Slack・Discordの場合はこちらの公開グループに参加してShoya YasudaまでDMをください!


※当ブログに関することは何でもご相談・ご依頼可能です。

この記事を書いた人
Yasuda

博士(理学)。専門は免疫細胞、数理モデル、シミュレーション。米国、中国で研究に携わった。遺伝的アルゴリズム信者。物価上昇のため半額弁当とともに絶滅寸前。

タイトルとURLをコピーしました