Yii2 設計模式——單例模式

單例模式

模式定義

單例模式確保一個類只有一個實例,並提供一個全局訪問點。當現實中只需要一個對象,或者爲了節省系統資源,又或者是爲了共享數據的時候可以使用單例模式。

代碼實現

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

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容器的設計是必然的、合理的。

應用舉例

在 Yii.php 中:

只在入口腳本require '…/Yii.php’時創建一個Container實例 Yii::$container = new yii\di\Container();

在Yii框架中,除了少數情況,都是通過Yii::createObject()來創建類的實例

而在Yii::createObject()中始終用到一個單例Yii::$container

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章