Yii2之組件的注冊與創建


  今天本來打算研究一下yii2.0AR模型的實現原理,然而,計划趕不上變化,突然就想先研究一下yii2.0的數據庫組件創建的過程。通過對yii源碼的學習,了解了yii組件注冊與創建的過程,並發現原來yii組件注冊之后並不是馬上就去創建的,而是待到實際需要使用某個組件的時候再去創建對應的組件實例的。本文大概記錄一下這個探索的過程。

  要了解yii組件的注冊與創建,當然要從yii入口文件index.php說起了,整個文件代碼如下:

<?php
defined('YII_DEBUG') or define('YII_DEBUG', true);
defined('YII_ENV') or define('YII_ENV', 'dev');

require(__DIR__ . '/../../vendor/autoload.php');
require(__DIR__ . '/../../vendor/yiisoft/yii2/Yii.php');
require(__DIR__ . '/../../common/config/bootstrap.php');
require(__DIR__ . '/../config/bootstrap.php');

$config = yii\helpers\ArrayHelper::merge(
    require(__DIR__ . '/../../common/config/main.php'),
    require(__DIR__ . '/../../common/config/main-local.php'),
    require(__DIR__ . '/../config/main.php'),
    require(__DIR__ . '/../config/main-local.php')
);

(new yii\web\Application($config))->run();

可以看到入口文件引入了幾個配置文件,並將所有配置文件的內容都合並到$config這個配置數組中,然后使用這個配置數組作為參數去創建一個應用實例。若將這個配置數組打印出來,就會看到,“components”下標對應的元素包含了yii組件的參數信息(這里只截圖一小部分):

這些組件的信息是在引入進來的幾個配置文件中配置的,Yii組件就是使用這些參數信息進行注冊與創建的。

  接下來就進入yii\web\Application類的實例化過程了,yii\web\Application類沒有構造函數,但是它繼承了\yii\base\Application類:

所以會自動執行\yii\base\Application類的構造函數:

public function __construct($config = [])
{
    Yii::$app = $this;
    static::setInstance($this);

    $this->state = self::STATE_BEGIN;

    $this->preInit($config);

    $this->registerErrorHandler($config);

    Component::__construct($config);
}

這里要順便說一下預初始化方法preInit(),它的代碼如下:

public function preInit(&$config)
{
    /* 此處省略對$config數組的預處理操作代碼 */

    // merge core components with custom components
    foreach ($this->coreComponents() as $id => $component) {
        if (!isset($config['components'][$id])) {
            $config['components'][$id] = $component;
        } elseif (is_array($config['components'][$id]) && !isset($config['components'][$id]['class'])) {
            $config['components'][$id]['class'] = $component['class'];
        }
    }
}

  這個函數對傳遞給構造函數的配置數組$config進行了一些預處理操作(這里省略了),最后使用coreComponents()方法返回的數組對$config數組進行了完善,coreComponents()方法是這樣的:

public function coreComponents()
{
    return [
        'log' => ['class' => 'yii\log\Dispatcher'],
        'view' => ['class' => 'yii\web\View'],
        'formatter' => ['class' => 'yii\i18n\Formatter'],
        'i18n' => ['class' => 'yii\i18n\I18N'],
        'mailer' => ['class' => 'yii\swiftmailer\Mailer'],
        'urlManager' => ['class' => 'yii\web\UrlManager'],
        'assetManager' => ['class' => 'yii\web\AssetManager'],
        'security' => ['class' => 'yii\base\Security'],
    ];
}

  其實就是一些核心組件的配置,也就是說這些組件是可以不需要我們在配置文件中配置的,yii會自動進行注冊。

  好了,回到\yii\base\Application類的構造函數,這個函數最后調用了\yii\base\Component類的構造函數,但\yii\base\Component類是沒有構造函數的,不過它繼承了\yii\base\Object類:

所以也自動執行了\yii\base\Object類的構造函數:

public function __construct($config = [])
{
    if (!empty($config)) {
        Yii::configure($this, $config);
    }
    $this->init();
}

這里主要是調用了\yii\BaseYii類的靜態方法configure()

public static function configure($object, $properties)
{
    foreach ($properties as $name => $value) {
        $object->$name = $value;
    }

    return $object;
}

這個方法就是循環入口文件(new yii\web\Application($config))->run();中的$config數組(這個數組的結構參見本文第一個截圖),以數組鍵名作為對象屬性名,對應的鍵值作為對象屬性值進行賦值操作。所以當循環到組件配置參數的時候是這樣子的:$object->components = $value$value為所有組件的配置數組),也就是對$objectcomponents屬性進行賦值操作,那這個$object是哪個類的對象呢?回想最初調用的源頭,其實它就是入口文件中需要進行實例化的\yii\web\Application類的對象啊。然而,這個類和它的祖先類都沒有components這個成員變量啊,不急,又要進行一番繼承套路了,順着yii\web\Application類的繼承關系一層一層往上找可以發現\yii\web\Application類最終也繼承了\yii\base\Object類,\yii\base\Object類是支持屬性的,所以yii\web\Application類也支持屬性(關於屬性,可以參考我的另一篇博文:yii2之屬性),當賦值操作找不到components成員變量時會調用setComponents()方法,又去找這個方法的所在,終於在它的祖先類\yii\di\ServiceLocator中找到了setComponents()方法,沒錯,對應用實例的components屬性進行賦值操作其實就是調用這個方法!

  好了,現在就來看看setComponents()這個方法到底干了啥:

public function setComponents($components)
{
    foreach ($components as $id => $component) {
        $this->set($id, $component);
    }
}

其實很簡單,就是循環各個組件的配置數組,調用set()方法,set()方法如下:

public function set($id, $definition)
{
    unset($this->_components[$id]);

    if ($definition === null) {
        unset($this->_definitions[$id]);
        return;
    }

    if (is_object($definition) || is_callable($definition, true)) {
        // an object, a class name, or a PHP callable
        $this->_definitions[$id] = $definition;
    } elseif (is_array($definition)) {
        // a configuration array
        if (isset($definition['class'])) {
            $this->_definitions[$id] = $definition;
        } else {
            throw new InvalidConfigException("The configuration for the \"$id\" component must contain a \"class\" element.");
        }
    } else {
        throw new InvalidConfigException("Unexpected configuration type for the \"$id\" component: " . gettype($definition));
    }
}

其實就是把組件配置存入$_definitions這個私有成員變量(即注冊),然后呢?然后就沒有下文了。。。

  搞了半天,原來yii創建應用實例的時候只是進行組件的注冊,並沒有實際創建組件,那么組件實例是什么時候進行創建的?在哪里進行創建的呢?別急。從上面推導的這個過程我們知道\yii\di\ServiceLocator類是\yii\web\Application類的祖先類,所以其實yii的應用實例其實就是一個服務定位器,比如我們想訪問數據庫組件的時候,我們可以這樣來訪問:Yii::$app->db,這個Yii::$app就是yii應用實例,也就是\yii\web\Application類的實例,但是\yii\web\Application類和它的父類、祖先類都找不到db這個屬性啊。哈哈,別忘了,php讀取不到類屬性的時候會調用魔術方法__get(),所以開始查找\yii\web\Application繼承關系最近的祖先類中的__get()方法,最后在\yii\di\ServiceLocator類中找到了,也就是說,Yii::$app->db最終會調用\yii\di\ServiceLocator類中的__get()方法:

public function __get($name)
{
    if ($this->has($name)) {
        return $this->get($name);
    } else {
        return parent::__get($name);
    }
}

__get()方法首先調用has()方法(這個不再貼代碼了)判斷組件是否已注冊,若已注冊則調用get()方法:

public function get($id, $throwException = true)
{
    if (isset($this->_components[$id])) {
        return $this->_components[$id];
    }

    if (isset($this->_definitions[$id])) {
        $definition = $this->_definitions[$id];
        if (is_object($definition) && !$definition instanceof Closure) {
            return $this->_components[$id] = $definition;
        } else {
            return $this->_components[$id] = Yii::createObject($definition);
        }
    } elseif ($throwException) {
        throw new InvalidConfigException("Unknown component ID: $id");
    } else {
        return null;
    }
}

其中私有成員變量$_components是存儲已經創建的組件實例的,若發現組件已經創建過則直接返回組件示例,否則使用$_definitions中對應組件的注冊信息,調用\yii\BaseYii::createObject()方法進行組件創建,這個方法最終會調用依賴注入容器\yii\di\Containerget()方法,接着就是依賴注入創建對象的過程了,關於這個過程已經在我的上一篇博文中講解過了,可以參考一下:yii2之依賴注入與依賴注入容器

  好了,yii組件注冊與創建的整個過程就是這樣的。最后總結一下,其實yii創建應用實例的時候只是進行了各個組件的注冊,也就是將組件的配置信息存入\yii\di\ServiceLocator類的私有成員變量$_definitions中,並沒有進行實際創建,等到程序運行過程中真正需要使用到某個組件的時候才根據該組件在$_definitions中保存的注冊信息使用依賴注入容器\yii\di\Container進行組件實例的創建,然后把創建的實例存入私有成員變量$_components,這樣下次訪問相同組件的時候就可以直接返回組件實例,而不再需要執行創建過程了。yii的這個組件注冊與創建機制其實是大有裨益的,試想一下,如果在應用實例創建的時候就進行所有組件的創建,將會大大增加應用實例創建的時間,用戶每次刷新頁面都會進行應用實例的創建的,也就是說用戶每刷新一次頁面都很慢,這用戶體驗就很不好了,而且很多情況下有很多組件其實是沒有使用到的,但是我們還是花了不少時間去創建這些組件,這是很不明智的,所以yii的做法就是:先把組件參數信息保存起來,需要使用到哪些組件再去創建相應的實例,大大節省了應用創建的時間,同時也節省了內存,這種思路是很值得我們學習的!

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM