laravel學習之IOC容器分析(一)


在入口文件里可以看到

$app = require_once __DIR__.'/../bootstrap/app.php';

$app這個是laravel的全局變量,在里面可以看到實例化了這個類

$app = new Illuminate\Foundation\Application(
    realpath(__DIR__.'/../')
);

里面的構造函數做了4件事

1,注冊基本的綁定,把當前類的對象綁定到app和Illuminate\Container\Container中

$this->registerBaseBindings();

2,注冊服務提供器,里面注冊了EventServiceProvider和RoutingServiceProvider,詳細的沒研究過,以后有機會在研究

$this->registerBaseServiceProviders()

3,注冊別名

$this->registerCoreContainerAliases();

4,綁定一些項目內有可能用到的路徑

if ($basePath) {
    $this->setBasePath($basePath);
}

 

 

接下來就一個一個的分析一下里面是用來做什么的

第一件事:

protected function registerBaseBindings()
    {
        static::setInstance($this);//獲取當前類的實例,規定了要實現ContainerContract接口的類才可以實例化

        $this->instance('app', $this);

        $this->instance('Illuminate\Container\Container', $this);
    }

第一句

public static function setInstance(ContainerContract $container)
    {
        static::$instance = $container;
    }

就把當前類的對象賦值給當前類的$instance變量,方便以后調用

剩下的就是調用instance方法來綁定對象到容器了

看看laravel是怎么綁定對象到容器的

public function instance($abstract, $instance)
    {
        $abstract = $this->normalize($abstract);//如果是字符串,就把前面的斜線去掉
if (is_array($abstract)) { list($abstract, $alias) = $this->extractAlias($abstract); $this->alias($abstract, $alias); }//如果是數組,例如$this->instance(['app'=>'a'],$this),就存進$this->aliases中,變成$this->aliases['app'] = 'a';沒用過這個綁定方法,查看其他資料看到的 unset($this->aliases[$abstract]); $bound = $this->bound($abstract);//返回一個布爾值,用於下面判斷是否執行回調函數 $this->instances[$abstract] = $instance;//這里就是把變量綁定到instance數組中,例如傳進來的是($app,$this),所以綁定成$this->instance['app'] = new a();a代表傳進來的類 if ($bound) {//如果為真,就調用$this->reboundCallback[]里有沒有回調函數,有的話就調用 $this->rebound($abstract); } }

就是這樣綁定了兩個變量到容器中了

第二件事:

protected function registerBaseServiceProviders()
    {
        $this->register(new EventServiceProvider($this));

        $this->register(new RoutingServiceProvider($this));
    }

注冊了兩個服務提供器,一個是事件服務提供器,一個是路由服務提供器

先看看register方法是做什么的 

public function register($provider, $options = [], $force = false)
    {
        if (($registered = $this->getProvider($provider)) && ! $force) {
            return $registered;
        }
if (is_string($provider)) { $provider = $this->resolveProviderClass($provider); } $provider->register();//注冊provider
foreach ($options as $key => $value) { $this[$key] = $value; } $this->markAsRegistered($provider);//記錄provider為已注冊
if ($this->booted) { $this->bootProvider($provider); } return $provider; }

首先判斷里面有沒有曾經注冊過,有的話直接返回

分析一下getProvider()方法

public function getProvider($provider)
    {
        $name = is_string($provider) ? $provider : get_class($provider);

        return Arr::first($this->serviceProviders, function ($key, $value) use ($name) {
            return $value instanceof $name;
        });
    }
public static function first($array, callable $callback = null, $default = null)
{
    if (is_null($callback)) {
        return empty($array) ? value($default) : reset($array);
    }

    foreach ($array as $key => $value) {
        if (call_user_func($callback, $key, $value)) {
            return $value;
        }
    }

    return value($default);
}

這里Arr::first主要是循環$this->serviceProviders這個數組,在里面找到已經注冊過的提供器,即如果$this->serviceProviders的鍵值是當前服務提供器的實例,就返回一個布爾值,然后register方法里面返回當前服務提供器的實例,找不到的話執行下面的代碼

 

if (is_string($provider)) {
            $provider = $this->resolveProviderClass($provider);
        }
$provider->register();//注冊provider

如果是字符串,直接實例化,然后調用這個類重寫之后的register()方法

$this->markAsRegistered($provider);//記錄provider為已注冊

里面調用了剛注冊了的event服務提供器里的fire方法,以后有機會在研究,然后就把當前類記錄為已注冊,就是把當前服務提供器的類名放進loadedProviders數組里

$this->serviceProviders[] = $provider;
$this->loadedProviders[$class] = true;

然后就調用提供器里的boot方法

if ($this->booted) {
            $this->bootProvider($provider);
        }

里面用了call方法來解決boot里面的依賴問題,laravel學習之IOC容器分析(二)在這里面解釋了,然后這樣就把一個服務提供器注冊成功了。

剩下的第三件事和第四件事就是把別名數組和laravel里面要用到的路徑注入到容器中。

到了這里,$app這個全局變量就完成了最基本的工作。

完....


免責聲明!

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



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