Yii2設計模式——單例模式

米粒人生發表於2018-12-19

應用舉例

在Yii.php中:

require __DIR__ . `/BaseYii.php`;

// Yii框架的幫助類,提供框架基本的功能
class Yii extends yiiBaseYii
{
}

spl_autoload_register([`Yii`, `autoload`], true, true);
Yii::$classMap = require __DIR__ . `/classes.php`;
// 只在入口指令碼require `../Yii.php`時建立一個Container例項
Yii::$container = new yiidiContainer();

在BaseYii中相關的部分是,

<?php

class BaseYii
{
    //Yii類維持一個DI Container的單例
    public static $container;

    //...

    // 在Yii框架中,除了少數情況,都是通過Yii::createObject()來建立類的例項
    // 而在Yii::createObject()中始終用到一個單例Yii::$container
    public static function createObject($type, array $params = [])
    {
        if (is_string($type)) {
            return static::$container->get($type, $params);
        } elseif (is_array($type) && isset($type[`class`])) {
            $class = $type[`class`];
            unset($type[`class`]);
            return static::$container->get($class, $params, $type);
        } elseif (is_callable($type, true)) {
            return static::$container->invoke($type, $params);
        } elseif (is_array($type)) {
            throw new InvalidConfigException(`Object configuration must be an array containing a "class" element.`);
        }

        throw new InvalidConfigException(`Unsupported configuration type: ` . gettype($type));
    }

    //...
}

這裡使用了單例模式。

單例模式

模式定義

單例模式確保一個類只有一個例項,並提供一個全域性訪問點。當現實中只需要一個物件,或者為了節省系統資源,又或者是為了共享資料的時候可以使用單例模式

程式碼實現

我們先來看看單例模式的標準實現:

final class Singleton
{    
    /**
     * @var Singleton
     * 維持一個對自身的引用,並保證其唯一性
     */
    private static $instance;

    
    // 獲取例項唯一的入口
    public static function getInstance(): Singleton
    {
        if (null === static::$instance) {
            static::$instance = new static();
        }

        return static::$instance;
    }
    
    // 不允許通過new的方式產生,只能通過Singleton::getInstance()方法
    private function __construct()
    {
    }
    
    // 也不允許clone()方法,此方法也會產生一個新的例項
    private function __clone()
    {
    }
    
    // 也不允許反序列化,因為反序列化也會產生一個新的例項
    private function __wakeup()
    {
    }
}

單例模式不允許產生單例的類被繼承,不允許通過new方式產生,除了規定的getInstance()方法,別的例項化的途徑基本被堵死。而在類的內部維持一個對自身的引用,並保證其是唯一的。

單例模式估計是所有涉及模式中最簡單的了,在PHP和Yii中很少見到直接這麼使用的,更多的是其變化的形式。

Yii的單例模式

Yii使用單例的場景非常多,比如請求開始建立的Application,Yii,Request,Response等物件功能都十分豐富且開銷也很大,維持一個單例就可供請求的整個生命週期使用。在請求開始即建立,請求結束自行銷燬,中間不銷燬也不建立。這些物件使用了單例沒有疑問,但是這些單例的產生、管理和使用卻是有不同講究的。

物件如何建立又如何維護,恐怕任何一個PHP框架都繞不開這個問題。Yii2採用服務定位器和依賴注入容器來提供大部分物件。在容器中使用單例好處是非常明顯的。至少可以表現在節省記憶體和公用元件方面。

節省記憶體

Yii::$container 在記憶體中僅有一份,所有使用DI容器的場合(Application/Module等)都用到這個DI容器。 這就節省了大量的記憶體空間和反覆構造例項的時間。

共用元件

更為重要的是,DI容器的單例化,使得Yii不同的模組共用元件成為可能。 可以想像,由於共用了DI容器,容器裡面的內容也是共享的。因此,你可以在A模組中改變某個元件的狀態,而B模組中可以瞭解到這一狀態變化。 但是,如果不採用單例模式,而是每個模組(Application/Module)都維護一個自己的DI容器, 要實現這一點難度會大得多。所以,這種共享DI容器的設計是必然的、合理的。

相關文章