LoginSignup
8
7

More than 5 years have passed since last update.

singleton, multitonパターンの実装でみるPHPの進化

Posted at

Singleton、Multitonパターンを実装することでPHPの進化を体感してみる。

PHP5.3

影響のあった新機能

コードと解説

遅延静的束縛の実装により、継承が行われたとしても親クラスとは別のインスタンスとしてSingletonパターンを構築することが可能となりました。
PHP5.2までで自クラスを表していたself::はself::を記述したクラスとして振る舞うため、継承先でも継承元のクラスを返す問題を抱えていました。

ここで着目すべきは次の2つのコードです。

static::$instance = new static();

この通り、staticキーワードのみで適切にインスタンスを作成できるようになりました。

$class_path = get_called_class();
static::$instance[$class_path] = new static();

ですが、multitonインスタンスでは、自クラスを表現するためにget_called_class()を呼び出す必要があります。
これは::classといった形でクラス名およびクラスのフルパスを得る事が出来ないためです。

php5.3
<?php
/**
 * Singleton
 */
class Singleton {
    static $instance = null;

    private final function __construct()
    {}

    public static function getInstance()
    {
        if (is_null(static::$instance)) {
            static::$instance = new static();
        }
        return static::$instance;
    }
}

/**
 * Multiton
 */
class Multiton {
    static $instance = array();

    private final function __construct()
    {}

    public static function getInstance()
    {
        $class_path = get_called_class();
        if (!isset(static::$instance[$class_path])) {
            static::$instance[$class_path] = new static();
        }
        return static::$instance[$class_path];
    }
}

/**
 * Use Singleton
 */
class SingleInstance extends Singleton {}

/**
 * Use Multiton
 */
class MultiInstance extends Multiton {}

PHP5.4

影響のあった新機能

コードと解説

mix inが可能となるtraitが実装されました。
SingletonやMultitonパターンを"特性"として切り出すことにより、継承ツリーにあまり影響を与えない形で柔軟に再利用を行うことができるようになりました。

デザインパターン自体には影響を与えてはいませんが、配列の省略表記もできるようになりました。
僅かですが、記述がスムーズになりました。

php5.4
<?php
/**
 * Singleton
 */
trait SingletonTrait {
    static $instance = null;

    private final function __construct()
    {}

    public static function getInstance()
    {
        if (is_null(static::$instance)) {
            static::$instance = new static();
        }
        return static::$instance;
    }
}

/**
 * Multiton
 */
trait MultitonTrait {
    static $instance = [];

    private final function __construct()
    {}

    public static function getInstance()
    {
        $class_path = get_called_class();
        if (!isset(static::$instance[$class_path])) {
            static::$instance[$class_path] = new static();
        }
        return static::$instance[$class_path];
    }
}

/**
 * Use Singleton
 */
class SingleInstance {
    use SingletonTrait;
}

/**
 * Use Multiton
 */
class MultiInstance {
    use MultitonTrait;
}

PHP5.5

影響のあった新機能

コードと解説

::class によるクラス名の解決が可能になったため、ついにPHP5.3から続いていた、get_called_classを使用しなくても済むようになりました。

php5.4
$class_path = get_called_class();
static::$instance[$class_path] = new static();

PHP5.4までは上記のように記述していたコードが、PHP5.5からは次のようにシンプルになります。

php5.5
if (!isset(static::$instance[static::class])) {
    static::$instance[static::class] = new static();
}
php5.5
/**
 * Singleton
 */
trait Singleton {
    static $instance = null;

    private final function __construct()
    {}

    public static function getInstance()
    {
        if (is_null(static::$instance)) {
            static::$instance = new static();
        }
        return static::$instance;
    }
}

/**
 * Multiton
 */
trait Multiton {
    static $instance = [];

    private final function __construct()
    {}

    public static function getInstance()
    {
        if (!isset(static::$instance[static::class])) {
            static::$instance[static::class] = new static();
        }
        return static::$instance[static::class];
    }
}

/**
 * Use Singleton
 */
class SingleInstance {
    use SingletonTrait;
}

/**
 * Use Multiton
 */
class MultiInstance {
    use MultitonTrait;
}

PHP7.0

影響のあった新機能

コードと解説

これもデザインパターンには影響はありませんが、記述が劇的に容易になる演算子が追加されました。

php5.5
if (!isset(static::$instance[static::class])) {
    static::$instance[static::class] = new static();
}
return static::$instance[static::class];

PHP5.6までは上記のように4行で記述していたコードが、PHP7.0からは次のように1行で完結します。

php7.0
return static::$instance[static::class] ?? static::$instance[static::class] = new static();
php7.0
/**
 * Singleton
 */
trait Singleton {
    static $instance = null;

    private final function __construct()
    {}

    public static function getInstance()
    {
        return static::$instance ?? static::$instance = new static();
    }
}

/**
 * Multiton
 */
trait Multiton {
    static $instance = [];

    private final function __construct()
    {}

    public static function getInstance()
    {
        return static::$instance[static::class] ?? static::$instance[static::class] = new static();
    }
}

/**
 * Use Singleton
 */
class SingleInstance {
    use SingletonTrait;
}

/**
 * Use Multiton
 */
class MultiInstance {
    use MultitonTrait;
}

終わりに

このようにPHPはより問題の解決を容易にするよう飽くなき進化を続けています。
この記事を読まれて、PHPの進化に興味を持たれた方はぜひ、PHP: 付録 - Manualから、各バージョンごとのPHPの新機能を読んでみてください。

最新のPHPについての記述はPHP 7.0.x から PHP 7.1.x への移行(2017/03/19時点)にあります。
このページにある、新機能を読めば新しいPHPの事をより一層理解することができるでしょう。

8
7
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
8
7