Yii使用單例的場景非常多,比如請求開始創建的Application,Yii,Request,Response等對象功能都十分豐富且開銷也很大,維持一個單例就可供請求的整個生命周期使用。在請求開始即創建,請求結束自行銷毀,中間不銷毀也不創建。這些對象使用了單例沒有疑問,但是這些單例的產生、管理和使用... ...
應用舉例
在Yii.php中:
require __DIR__ . '/BaseYii.php';
// Yii框架的幫助類,提供框架基本的功能
class Yii extends \yii\BaseYii
{
}
spl_autoload_register(['Yii', 'autoload'], true, true);
Yii::$classMap = require __DIR__ . '/classes.php';
// 只在入口腳本require '../Yii.php'時創建一個Container實例
Yii::$container = new yii\di\Container();
在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容器的設計是必然的、合理的。