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