はじめに
様々な言語で「デザインパターン」の本が世の中にありますが、筆者個人の経験では
いまいちピンとこない例
いまいちピンとこないコード
で説明されてることが多く、
結局これっていつ使うの?
という疑問に答えるには仕事仲間等との議論をしないと
辿り着けないことが多々ありました。
そこで特に「ゲーム開発ではどう使うか?」にフォーカスを当てて、実践的な例を交えて
デザインパターンの説明の需要があると思い記事を作りました。
デザインパターンを学ぶ理由
デザインパターンを学ぶ理由としては
- 車輪の再発明の防止
- 長文で読みにくいコード(可読性の低いコード)を減らす
- コードを疎結合にして変更に強くなる(変更時のコスト・変更箇所を減らす)
- モジュールとして使いまわせるように、コードの再利用性を高める
といった効果を期待できます。
対象読者
Unity 全くの初心者(インストールしただけで触ったことがないような方)はお断りです。
最低限以下のことは理解・経験を積んでおくことが必須になります。
- MonoBehaviour 継承クラスでコードを書いたことがある
- C# のピュアクラスを用いた自作クラスを作ったことがある
- クラスの継承という概念は知っている
そのため、脱・初心者
中級者へのステップアップ
として デザインパターンを学ぶ
のが良いと思います。
デザパタ記事リンク
生成系
構造系
様態・ふるまい系
- Chain of Responsibility パターン
- Command パターン
- Interpreter パターン
- Iterator パターン
- Mediator パターン
- Memento パターン
- Observer パターン
- State パターン
- Strategy パターン
- TemplateMethod パターン
- Visitor パターン
Builder パターンについて
Builder パターンとは 複雑なものから成るObjectを構成する
デザインパターンです。
特に 構成素材
と 構成過程
が重要、柔軟にしたい場合に利用されます。
・・・
AbstractFactory と何が違うのか?とお考えの読者の方、 非常に鋭い
です。
Abstract Factoryとの違い
簡単に答えるなら AbstractFactory は 0 -> 1 を行うもの
、Builder は 複数のものをまとめあげるもの
といった形です。
特に、Factory に関しては生成プロセスに対して外側からは介入することができません。一方Builder は生成過程に介入しやすいため、単純なAbstractFactoryに比べたら柔軟性が上がっています。
ゲームでの利用例
今までは敵の生成では EnemyFactory
を用意して生成することで問題ありませんでした。
しかし敵でもバリエーションを出したい状況はかなりの高確率で発生します。
たとえば「スケルトン(骸骨)」の敵を考えてみましょう。
UnityのAssetStore を検索すると以下のようなアセットが公開されています。
上記アセットからも分かる通り「スケルトン」一つとっても武器だけでも「なし」「刀」「弓」、防具も「あり」「なし」の組み合わせがあるため
全部で6通りのスケルトンを作ることができます。
このように考えると SkeletonFactory
を作ろうとしても工場の種類の組合せ爆発が起こります。
- NormalSkeletonFacroy
- WarriorSkeletonFacroy
- ArcherSkeletonFactory
- ArmoredSkeletonFacroy
- ArmoredWarriorSkeletonFacroy
- ArmoredArcherSkeletonFactory
名前も長いし、同じようなFactory を作らなきゃいけないですし、無駄にコードも増えて保守性も下がります。
そこで以下のように考えてみましょう。
- 「スケルトン」「各種武器」「防具」はそれぞれの工場から作成します、
-
SkeletonBuilder
は工場から渡された素材を組み合わせて敵としてのスケルトンを出荷するのみに特化させます。 - このときBuilder には 武器・防具は渡されない場合もあります
このときSkelteonBuilder は以下のようなAPIを持っています。
public abstract class BaseSkeletonBuilder
{
public abstract void SetSkeleton(Skeleton body);
public abstract void SetWeapon(BaseWeapon weapon);
public abstract void SetArmor(BaseArmor armor);
public abstract Skeleton Build()
}
大元となるSkeleton は必須ではありますが、そのほかはオプションで あってもなくても良い
というのが面白いところです。
Nullならセットしないだけで、必要に応じて適切にオプションのオブジェクトを登録するだけで、あとはBuilderでBuild()するだけで敵が作られるというカラクリです。
また、今は刀と弓ですが、「魔法使い」や「神官」などスケルトンのジョブが増えた時も「渡す武器・防具」を変えるだけで対応可能ですね。
まとめ
Builder はFactory などから提供される素材を組み合わせて新しくオブジェクトを作るときに便利なデザインパターンです。
Factory よりも 素材
や 生成過程
を柔軟にしたいときに大きく効果を発揮します。
参考
ピクトグラム: https://icooon-mono.com/license/