談laravel容器延遲加載以及auth擴展

昨天按照手冊教程,動手寫一個Auth擴展,按照包獨立性的原則,我不但願將Auth::extend()這種方法寫在 start.php 中,毫無疑問,我選擇了在服務提供器register()方法中註冊擴展驅動。然而,事與願違……php

http://www.zroran.com/it/php/laravel/10.htmlhtml

###發現問題laravel

當我在 LoauthServiceProvider 中這樣寫的時候:數組

public function register()
{
	//
    \Auth::extend('loauth',function($app){});
}

報錯app

Call to undefined method Illuminate\Support\Facades\Auth::extend()

###尋找緣由框架

當時就納悶了,找緣由,懷疑是Auth沒註冊?檢查發現註冊了,由於在路由中可使用;php artisan clear-compiled 沒用;百思不得其解,甚至懷疑是我不當心修改了核心類,還從新下載了一次laravel包,問題依舊。ide

折騰了一夜,最終我把目光鎖定在 AuthServiceProvider$defer 屬性。學習

根據手冊以及註釋,咱們得知 $defer 屬性是用來延遲加載該服務提供器,說直白點就是延遲執行 register() 方法,只須要配合provides()方法便可實現。舉個例子:this

public function provides()
{
	return array('auth');
}

這個是 AuthServiceProvider 裏的方法,當框架初始化的時候,會依次加載服務提供器,若是發現這個服務提供器protected $defer=true 那麼就會調用它的 provides() 方法,其返回的數組包含須要延遲加載的服務名稱,這樣當咱們在路由、控制器或者其餘地方調用 Auth::METHOD() 的時候,纔會去調用提供器的 register() 方法。code

###肯定癥結 那麼問題來了,既然是被動延遲加載,也就是說當我調用Auth類方法時應該會自動實例化Auth類啊,爲何我在LoauthServiceProvider中調用的時候卻提示方法不存在,可是在路由中卻能夠呢。

我猜想是由於優先級的問題,可能在框架註冊 LoauthServiceProvider::register() 的時候,Auth 尚未標記爲延遲加載,這就形成了一個前後問題,任何即時加載的服務提供器都沒法在register方法中調用延遲加載的服務

通過研究,順利在覈心代碼中找到證據 Illuminate\Foundation\ProviderRepository

public function load(Application $app, array $providers)
{
	//...省略

	// We will go ahead and register all of the eagerly loaded providers with the
	// application so their services can be registered with the application as
	// a provided service. Then we will set the deferred service list on it.
	foreach ($manifest['eager'] as $provider)
	{
		$app->register($this->createProvider($app, $provider));
	}
	//延遲加載標記在即時加載服務以後
    $app->setDeferredServices($manifest['deferred']);
}

###解決之道

雖然發現了問題所在,但並不表明問題就解決了,修改核心代碼不是個明智的選擇,因此只能在咱們本身的包裏想辦法咯,一個解決方案以下:

public function register()
{
	//
    $authProvider = new \Illuminate\Auth\AuthServiceProvider($this->app);
    $authProvider->register();
    \Auth::extend('loauth',function($app){});
}

既然auth還未註冊,那麼咱們手動調用它的register方法幫它註冊。

###留言 研究框架也是個學習的過程,或許你還有更好的方法,歡迎留言分享 ^v^

相關文章
相關標籤/搜索