0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Matplotlibでの3D分子構造アニメーション - 化学反応のダイナミクスを視覚的に理解

Posted at

はじめに

分子構造の可視化は、化学や生物学の分野で重要な役割を果たしています。静的な2D図では表現しきれない分子の立体構造や動的な振る舞いを理解するためには、3Dアニメーションが非常に有効です。この記事では、Google Colaboratory(Google Colab)を使用して、PythonのMatplotlibライブラリによるメタン(CH₄)分子の3D構造アニメーション化を解説します。

Google Colabを使用することで、以下のような利点があります:

  1. 環境構築が不要:特別なセットアップなしで、ブラウザ上で直接コードを実行できます。
  2. 無料のGPUアクセス:複雑な計算が必要な場合、GPUを利用できます(今回のコードでは不要です)。
  3. 簡単な共有:作成したノートブックを他の人と簡単に共有できます。
  4. クラウド上での実行:ローカルマシンのリソースを気にせずに実行できます。

Google Colabの準備

  1. Google Colab にアクセスします。
  2. 新しいノートブックを作成します。
  3. ランタイムのタイプが「Python 3」になっていることを確認します。

必要なライブラリ

まず、以下のライブラリをインポートします。Google Colabでは、これらのライブラリは事前にインストールされているため、追加のインストール作業は必要ありません。

import numpy as np
import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d import Axes3D
import matplotlib.animation as animation
from IPython.display import HTML

各ライブラリの役割:

  • numpy: 数値計算のための基本ライブラリ
  • matplotlib.pyplot: プロットの作成と管理
  • mpl_toolkits.mplot3d: 3Dプロットの作成
  • matplotlib.animation: アニメーションの作成
  • IPython.display: IPython環境(Google Colabを含む)での表示用

メタン分子の構造定義

メタン分子は、1つの炭素原子を中心に4つの水素原子が正四面体構造で配置されています。この構造を以下のように定義します。

# メタン分子の構造を定義 (正四面体構造)
carbon_position = np.array([0, 0, 0])
hydrogen_positions = np.array([
    [1, 1, 1],
    [-1, -1, 1],
    [-1, 1, -1],
    [1, -1, -1]
]) / np.sqrt(3)  # 正規化

initial_positions = np.vstack((carbon_position, hydrogen_positions))

この定義では:

  • 炭素原子を原点 (0, 0, 0) に配置
  • 水素原子を正四面体の頂点に配置(正規化して単位長さにしています)

アニメーションの準備

アニメーションのパラメータを設定し、各フレームでの原子の位置を計算する関数を定義します。

# アニメーションのフレーム数
n_frames = 100

# 振動の振幅と周波数
amplitude = 0.05
frequency = 2 * np.pi / n_frames

# 各フレームの原子位置を計算
def calculate_positions(frame):
    vibration = amplitude * np.sin(frequency * frame + np.random.rand(*initial_positions.shape) * np.pi) * np.random.rand(*initial_positions.shape)
    vibration[0] *= 0.2  # 炭素原子の振動を抑える
    return initial_positions + vibration

この関数では:

  • 正弦波を使用して周期的な振動を生成
  • ランダム性を加えて自然な動きを表現
  • 炭素原子(中心原子)の振動を水素原子よりも小さくしています

描画関数の定義

原子を球体として描画する関数を定義します。

# 球体を描画する関数
def draw_sphere(ax, center, radius, color):
    u, v = np.mgrid[0:2*np.pi:20j, 0:np.pi:10j]
    x = center[0] + radius * np.cos(u) * np.sin(v)
    y = center[1] + radius * np.sin(u) * np.sin(v)
    z = center[2] + radius * np.cos(v)
    ax.plot_surface(x, y, z, color=color, alpha=0.8, shade=True)

この関数は:

  • 球面の座標を計算
  • plot_surface メソッドを使用して球体を描画
  • alpha パラメータで透明度を設定し、shade=True で陰影を付けています

アニメーション更新関数

各フレームでの描画を更新する関数を定義します。

# アニメーション更新関数
def update(frame):
    ax.clear()
    ax.set_xlim(-1, 1)
    ax.set_ylim(-1, 1)
    ax.set_zlim(-1, 1)
    ax.set_title("Methane (CH₄) Molecule Animation", fontsize=16)
    
    # 座標軸を表示
    ax.set_xlabel('X')
    ax.set_ylabel('Y')
    ax.set_zlabel('Z')
    
    # 背景色を設定
    ax.set_facecolor((0.95, 0.95, 1))
    
    # 原子位置の計算
    positions = calculate_positions(frame)
    
    # 結合を描画
    for i in range(1, 5):
        ax.plot(*zip(positions[0], positions[i]), color='gray', alpha=0.5, linewidth=3)
    
    # 原子を描画
    for pos, c, s, label in zip(positions, colors, sizes, labels):
        draw_sphere(ax, pos, s, c)
        ax.text(pos[0], pos[1], pos[2], label, color='red', fontsize=12, ha='center', va='center')
    
    # カメラアングルの調整
    ax.view_init(elev=20, azim=45 + frame)

    return ax

この関数では:

  • フレームごとにプロットをクリアし、再描画
  • 座標軸の範囲設定とラベル付け
  • 原子間の結合を線で表現
  • 各原子を球体として描画し、ラベルを付加
  • カメラアングルを動的に変更して回転効果を生成

アニメーションの作成と表示

最後に、アニメーションを作成し、HTML5ビデオとして表示します。

# プロットの設定
fig = plt.figure(figsize=(12, 12))
ax = fig.add_subplot(111, projection='3d')

# 原子の色とサイズを設定
colors = ['black', 'white', 'white', 'white', 'white']
sizes = [0.3, 0.2, 0.2, 0.2, 0.2]
labels = ['C', 'H', 'H', 'H', 'H']

# アニメーションの作成
anim = animation.FuncAnimation(fig, update, frames=n_frames, interval=50, blit=False)

# HTML5ビデオとしてアニメーションを表示
video = anim.to_html5_video()
HTML(video)

このコードでは:

  • 3Dプロットの図を作成
  • 原子の色、サイズ、ラベルを設定
  • FuncAnimation を使用してアニメーションを生成
  • アニメーションをHTML5ビデオに変換し、表示

結果と考察

このコードを実行すると、メタン分子の3Dアニメーションが表示されます。アニメーションでは以下の特徴が観察できます:

  1. メタン分子の正四面体構造
  2. 炭素原子(黒)と水素原子(白)の相対的な大きさ
  3. 分子内の原子の振動の様子
  4. 分子全体の3D構造とその対称性
  5. 分子の回転による様々な角度からの視点

このアニメーションにより、静的な2D表現では難しかった以下のような分子の特性を視覚的に理解することができます:

  • 分子の立体構造:正四面体構造が明確に見えます。
  • 原子間の相互作用:結合の柔軟性や分子全体の動きが表現されています。
  • 空間的配置:原子の3次元的な位置関係が直感的に理解できます。

Google Colabでの実行に関する注意点

  1. メモリ使用量:長時間の実行や大規模なデータセットを扱う場合、Google Colabのメモリ制限に注意してください。
  2. セッションの維持:長時間の非アクティブ状態ではセッションが切断される可能性があります。定期的に作業を保存しましょう。
  3. GPU加速:本コードではGPU加速は必要ありませんが、より複雑な計算を行う場合はGPUランタイムの使用を検討してください。

発展的な応用

このコードを基にして、さらに複雑な分子構造や化学反応のシミュレーションを行うことができます。例えば:

  1. 異なる分子の構造をモデル化し、比較する

    • 水分子(H₂O)や二酸化炭素(CO₂)など、異なる分子構造をモデル化し、メタンとの違いを視覚化する。
  2. 化学反応の過程をアニメーション化する

    • メタンの燃焼反応(CH₄ + 2O₂ → CO₂ + 2H₂O)のような化学反応を段階的にアニメーション化する。
  3. 温度変化による分子の振動の変化を表現する

    • 振動の振幅を温度のパラメータと連動させ、温度上昇に伴う分子運動の活発化を表現する。
  4. 複数の分子の相互作用をシミュレートする

    • 複数のメタン分子や他の分子との相互作用を表現し、気体や液体の性質を視覚化する。
  5. 電子雲の表現を追加する

    • 分子軌道理論に基づいて、電子雲の分布を半透明の表面として表現する。
  6. インタラクティブな要素を追加する

    • ユーザーが分子を回転させたり、振動の振幅を調整したりできるインタラクティブな機能を実装する。

まとめ

Google Colabを使用したMatplotlibによる3D分子構造のアニメーション化は、化学や生物学の概念を視覚的に理解する強力なツールとなります。環境構築の手間なく、ブラウザ上で直接コードを実行し結果を得られる利点は、特に教育現場や研究の初期段階で大きな価値があります。

この記事で紹介した手法を応用することで、様々な分子や化学プロセスを動的に表現し、より深い理解を得ることができるでしょう。さらに、Google Colabの共有機能を活用すれば、作成したアニメーションや解析結果を簡単に他の研究者と共有することもできます。

分子構造や化学反応の視覚化は、単に美しいビジュアルを作るだけでなく、複雑な概念の理解を助け、新しい洞察を得るための重要なツールとなります。このような技術を活用することで、化学教育の質を向上させ、研究者間のコミュニケーションを促進することができるでしょう。

最後に、このコードは基本的な実装例です。読者の皆さんがこれを基に、より複雑で興味深い分子モデルや化学現象の視覚化にチャレンジされることを期待しています。科学の世界をより身近に、より理解しやすくするための一助となれば幸いです。

実装(動作確認用)

google colab

import numpy as np
import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d import Axes3D
import matplotlib.animation as animation
from IPython.display import HTML

# メタン分子の構造を定義 (正四面体構造)
carbon_position = np.array([0, 0, 0])
hydrogen_positions = np.array([
    [1, 1, 1],
    [-1, -1, 1],
    [-1, 1, -1],
    [1, -1, -1]
]) / np.sqrt(3)  # 正規化

initial_positions = np.vstack((carbon_position, hydrogen_positions))

# アニメーションのフレーム数
n_frames = 100

# 振動の振幅と周波数
amplitude = 0.05
frequency = 2 * np.pi / n_frames

# 各フレームの原子位置を計算
def calculate_positions(frame):
    vibration = amplitude * np.sin(frequency * frame + np.random.rand(*initial_positions.shape) * np.pi) * np.random.rand(*initial_positions.shape)
    vibration[0] *= 0.2  # 炭素原子の振動を抑える
    return initial_positions + vibration

# プロットの設定
fig = plt.figure(figsize=(12, 12))
ax = fig.add_subplot(111, projection='3d')

# 原子の色とサイズを設定
colors = ['black', 'white', 'white', 'white', 'white']
sizes = [0.3, 0.2, 0.2, 0.2, 0.2]
labels = ['C', 'H', 'H', 'H', 'H']

# 球体を描画する関数
def draw_sphere(ax, center, radius, color):
    u, v = np.mgrid[0:2*np.pi:20j, 0:np.pi:10j]
    x = center[0] + radius * np.cos(u) * np.sin(v)
    y = center[1] + radius * np.sin(u) * np.sin(v)
    z = center[2] + radius * np.cos(v)
    ax.plot_surface(x, y, z, color=color, alpha=0.8, shade=True)

# アニメーション更新関数
def update(frame):
    ax.clear()
    ax.set_xlim(-1, 1)
    ax.set_ylim(-1, 1)
    ax.set_zlim(-1, 1)
    ax.set_title("Methane (CH₄) Molecule Animation", fontsize=16)
    
    # 座標軸を表示
    ax.set_xlabel('X')
    ax.set_ylabel('Y')
    ax.set_zlabel('Z')  # 修正済み: set_zabel から set_zlabel に変更
    
    # 背景色を設定
    ax.set_facecolor((0.95, 0.95, 1))
    
    # 原子位置の計算
    positions = calculate_positions(frame)
    
    # 結合を描画
    for i in range(1, 5):
        ax.plot(*zip(positions[0], positions[i]), color='gray', alpha=0.5, linewidth=3)
    
    # 原子を描画
    for pos, c, s, label in zip(positions, colors, sizes, labels):
        draw_sphere(ax, pos, s, c)
        ax.text(pos[0], pos[1], pos[2], label, color='red', fontsize=12, ha='center', va='center')
    
    # カメラアングルの調整
    ax.view_init(elev=20, azim=45 + frame)

    return ax

# アニメーションの作成
anim = animation.FuncAnimation(fig, update, frames=n_frames, interval=50, blit=False)

# HTML5ビデオとしてアニメーションを表示
video = anim.to_html5_video()
html = HTML(video)
display(html)

# 静止画も表示(確認用)
plt.close(fig)  # アニメーションのfigureを閉じる
fig_static = plt.figure(figsize=(12, 12))
ax_static = fig_static.add_subplot(111, projection='3d')
update(0)  # 最初のフレームを表示
plt.show()

print("上に表示されているのはメタン(CH₄)分子のアニメーションと静止画です。中心の黒い球が炭素原子、周囲の白い球が水素原子を表しています。")
0
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?