タイトル: staticクラスとシングルトンの違い
本稿ではstaticクラスとシングルトンの違いを説明します。どちらも「インスタンスが事実上1つしかない」状態を作るパターンですが、設計上の性質や扱いやすさに違いがあります。
※随時追加予定
本質的な違い
| 項目 | staticクラス | シングルトン |
|---|---|---|
| インスタンス | 持たない(メンバはすべてstatic) | 1つだけ持つ(インスタンス取得用メソッド経由) |
| 状態保持 | 静的フィールド | インスタンスフィールド |
| 引数として渡す | 不可(クラス自体は渡せない) | 可能(オブジェクトとして扱える) |
| インターフェース実装 | 不可(staticメソッドはインターフェース要件を満たせない) | 可能 |
| 継承・オーバーライド | 不可 | 可能 |
| 遅延初期化 | クラスロード時に固定 | 初回 getInstance() で生成可能 |
| テスト容易性 | 差し替え不可、モック困難 | DI(依存注入)で差し替え可能 |
| マルチインスタンス化 | 不可(1つの状態) | 設計上は1つだが、必要なら派生で複数可 |
シングルトンの利点(記事原文に基づく)
- シングルトンはオブジェクトとして扱うことが出来るので、あるメソッドの引数に渡してやることが出来る。
- シングルトンはインターフェースを利用できるのでモック化が楽。
結果として、シングルトンはテストやDI(依存注入)と相性が良いのが大きな強みです。staticクラスは「単純で書きやすい」が「テスト・差し替えに弱い」というトレードオフがあります。
コードでの違い(Java)
staticクラス(ユーティリティクラス)
|
public final class StringUtils { |
シングルトン
|
public class ConfigService { |
使い分けの目安
| 場面 | 推奨 |
|---|---|
| 状態を持たない純粋な関数群 | staticクラス(ユーティリティ) |
| システム全体で1つだけ持ちたい設定・キャッシュ | シングルトン(できればDIで管理) |
| テストでモックに差し替えたい | シングルトン or DI |
| インターフェースを介して使いたい | シングルトン |
| マルチスレッドで状態共有 | シングルトン(適切なロック設計) |
近年のベストプラクティス
自前で getInstance() を実装するシングルトンより、DIコンテナ(Spring、Guice、CDI等)にシングルトンスコープで管理させるのが現在の主流です。テスト時に容易にモックへ差し替えられ、依存関係も明示できます。
|
// Spring の例: @Service / @Component で自動的にシングルトン |
注意点
- シングルトンの濫用はグローバル状態を増やし、結合度が上がる
- マルチスレッドでの初期化:
synchronizedやdouble-checked locking、enum singletonパターン - テスト時のリセットがしにくい場合は、DI管理にする
- staticは継承できないため、機能拡張時に困ることがある
関連
- 親カテゴリ: シングルトン
- デザインパターン全般: デザインパターン
- 関連: オブジェクト指向言語共通