UnityのMonoBehaviourクラスをシングルトン化する

何を隠そう、ここ2年ほど仕事でUnityを使っているオイラです。

UnityでC#のScriptを作成しようとすると、デフォルトでMonoBehaviourクラスを継承したクラスが作成される。このMonoBehaviourクラスはUnityとやり取りするための様々な機能がすでに実装されていてとても便利。ただ、個々のGameObjectが独立して動作させるような時には自由度が高くて良いんだけど、この自由度が逆に管理しづらいこともあるのです。

例えば、ゲーム全体を一元管理する情報を扱うような場合、あるいは外部のアプリケーションやKinectみたいなセンサーとやり取りするインターフェイスを作る場合、そのクラスのインスタンスは唯一にしておきたい。知らないうちに複数出来ちゃったら大混乱だ。

ということで、MonoBehaviourクラスの便利さを保ったままシングルトンにする方法を探したら、良い方法を見つけた。

Unity Technologies Japanの元エバンジェリスト伊藤さんが翻訳したこの記事↓
http://warapuri.tumblr.com/post/28972633000/unity-50-tips

29、シングルトンは便利だ

こんな感じで自動的にシングルトンが作られるようにしておくといい。

public class Singleton<T> : MonoBehaviour where T : MonoBehaviour
{
   protected static T instance;
 
   /**
      Returns the instance of this singleton.
   */
   public static T Instance
   {
      get
      {
         if(instance == null)
         {
            instance = (T) FindObjectOfType(typeof(T));
 
            if (instance == null)
            {
               Debug.LogError("An instance of " + typeof(T) + 
                  " is needed in the scene, but there is none.");
            }
         }
 
         return instance;
      }
   }
}

シングルトンはParticleManager とか AudioManager とか GUIManager.とかマネージャーに役に立つ。

  • マネージャーではない固有インスタンス(プレイヤーとか)にシングルトンを使うのは止めよう。(以下略)
  • クラスの外側から利用されるために、staticなプロパティと関数を定義しておこう。こうしておくと、GameManager.Playerと書けるようになる(GameManager.Instance.playerという書き方ではなく)

あとはこれを継承して任意のクラスを作るだけ。例えばこんな感じ。

using UnityEngine;
 
public class GameManager : SingletonMonoBehaviour<GameManager> {
     
    public void Awake(){
        if(this != Instance){
            Destroy(this);
            return;
        }
 
        DontDestroyOnLoad(this.gameObject);//シーン遷移しても破棄されない設定
    }    
     
}

そして、同じような要件でシーン遷移後もMonoBehaviourクラスを破棄せずに存続させたい場合は、上記のようにAwake()メソッドでの初期化時に

 DontDestroyOnLoad(this.gameObject);

という記述を加えれば、Script(MonoBehaviourクラス)の親となるGameObjectがシーン遷移後も存続するようになる。

Unityは、ささっと動く物を作るには本当に便利だけど、ややカチッとしたものを作ろうとするとMonoBehaviourクラスの特殊さが壁になっちゃったりもする。
システム開発してた人間からすると、MVCにならない時点で頭を抱えちゃうんだけど、Unityに適した設計方法ってのをのんびり模索していきたい。上手く長所を活かせる方法が見つかるといいなぁ。

関連記事

OpenCV3.3.0でsfmモジュールのビルドに成功!

OpenCVの三角測量関数『cv::triangulatep...

COLMAP:オープンソースのSfM・MVSツール

ブログが1日ダウンしてました

UnityからROSを利用できる『ROS#』

Photogrammetry (写真測量法)

Kaolin:3Dディープラーニング用のPyTorchライブ...

UnityプロジェクトをGitHubで管理する

Mean Stack開発の最初の一歩

WordPressプラグインの作り方

機械学習で遊ぶ

Dlib:C++の機械学習ライブラリ

libigl:軽量なジオメトリ処理ライブラリ

WordPressのテーマを自作する

Kornia:微分可能なコンピュータービジョンライブラリ

DensePose:画像中の人物表面のUV座標を推定する

Multi-View Environment:複数画像から3...

Caffe:読みやすくて高速なディープラーニングのフレームワ...

Verilog HDL

サンプルコードにも間違いはある?

Twitter APIのPythonラッパー『python-...

trimesh:PythonでポリゴンMeshを扱うライブラ...

OpenCVのバージョン3が正式リリースされたぞ

Mitsuba 2:オープンソースの物理ベースレンダラ

Unityからkonashiをコントロールする

WordPressプラグインによるサイトマップの自動生成

ブログをGoogle App EngineからAmazon ...

CGレンダラ研究開発のためのフレームワーク『Lightmet...

UnityのTransformクラスについて調べてみた

WinSCP

PeopleSansPeople:機械学習用の人物データをU...

Mitsuba 3:オープンソースの研究向けレンダラ

Deep Fluids:流体シミュレーションをディープラーニ...

UnityのuGUIチュートリアル

OpenCVのfindEssentialMat関数を使ったサ...

Google App Engine上のWordPressでA...

Python.NET:Pythonと.NETを連携させるパッ...

UnityのGlobal Illumination

Unity MonoBehaviourクラスのオーバーライド...

pythonの機械学習ライブラリ『scikit-learn』

JavaScriptとかWebGLとかCanvasとか

Unreal Engineの薄い本

コメント