這篇文章主要介紹Laravel中的依賴注入和IoC的示例分析,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!
站在用戶的角度思考問題,與客戶深入溝通,找到曲麻萊網(wǎng)站設(shè)計(jì)與曲麻萊網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名注冊(cè)、網(wǎng)站空間、企業(yè)郵箱。業(yè)務(wù)覆蓋曲麻萊地區(qū)。依賴注入一詞是由 Martin Fowler 提出的術(shù)語,它是將組件注入到應(yīng)用程序中的一種行為。就像 Ward Cunningham 說的:
依賴注入是敏捷架構(gòu)中關(guān)鍵元素。
讓我們看一個(gè)例子:
class UserProvider{ protected $connection; public function __construct(){ $this->connection = new Connection; } public function retrieveByCredentials( array $credentials ){ $user = $this->connection ->where( 'email', $credentials['email']) ->where( 'password', $credentials['password']) ->first(); return $user; } }
如果你要測(cè)試或者維護(hù)這個(gè)類,你必須訪問數(shù)據(jù)庫(kù)的實(shí)例來進(jìn)行一些查詢。為了避免必須這樣做,你可以將此類與其他類進(jìn)行解耦 ,你有三個(gè)選項(xiàng)之一,可以將Connection
類注入而不需要直接使用它。
將組件注入類時(shí),可以使用以下三個(gè)選項(xiàng)之一:
class UserProvider{ protected $connection; public function __construct( Connection $con ){ $this->connection = $con; } ...
同樣,我們也可以使用 Setter 方法注入依賴關(guān)系:
class UserProvider{ protected $connection; public function __construct(){ ... } public function setConnection( Connection $con ){ $this->connection = $con; } ...
interface ConnectionInjector{ public function injectConnection( Connection $con ); } class UserProvider implements ConnectionInjector{ protected $connection; public function __construct(){ ... } public function injectConnection( Connection $con ){ $this->connection = $con; } }
當(dāng)一個(gè)類實(shí)現(xiàn)了我們的接口時(shí),我們定義了injectConnection
方法來解決依賴關(guān)系。
現(xiàn)在,當(dāng)測(cè)試我們的類時(shí),我們可以模擬依賴類并將其作為參數(shù)傳遞。每個(gè)類必須專注于一個(gè)特定的任務(wù),而不應(yīng)該關(guān)心解決它們的依賴性。這樣,你將擁有一個(gè)更專注和可維護(hù)的應(yīng)用程序。
如果你想了解更多關(guān)于 DI 的信息,Alejandro Gervassio 在 本系列 文章中對(duì)其進(jìn)行了廣泛而專業(yè)的介紹,所以一定要去讀這些文章。那么,什么又是 IoC 呢?IoC (控制反轉(zhuǎn))不需要使用依賴注入,但它可以幫助你有效的管理依賴關(guān)系。
Ioc 是一個(gè)簡(jiǎn)單的組件,可以更加方便地解析依賴項(xiàng)。你可以將對(duì)象形容為容器,并且每次解析類時(shí),都會(huì)自動(dòng)注入依賴項(xiàng)。
當(dāng)你請(qǐng)求一個(gè)對(duì)象時(shí), Laravel Ioc 在解決依賴關(guān)系的方式上有些特殊:
我們使用一個(gè)簡(jiǎn)單的例子,將在本文中改進(jìn)它。SimpleAuth
類依賴于FileSessionStorage
,所以我們的代碼可能是這樣的:
class FileSessionStorage{ public function __construct(){ session_start(); } public function get( $key ){ return $_SESSION[$key]; } public function set( $key, $value ){ $_SESSION[$key] = $value; } } class SimpleAuth{ protected $session; public function __construct(){ $this->session = new FileSessionStorage; } } //創(chuàng)建一個(gè) SimpleAuth $auth = new SimpleAuth();
這是一種經(jīng)典的方法,讓我們從使用構(gòu)造函數(shù)注入開始。
class SimpleAuth{ protected $session; public function __construct( FileSessionStorage $session ){ $this->session = $session; } }
現(xiàn)在我們創(chuàng)建一個(gè)對(duì)象:
$auth = new SimpleAuth( new FileSessionStorage() );
現(xiàn)在我想使用 Laravel Ioc 來管理這一切。
因?yàn)?code>Application 類繼承自Container
類,所以你可以通過App
門面來訪問容器。
App::bind( 'FileSessionStorage', function(){ return new FileSessionStorage; });
bind
方法第一個(gè)參數(shù)是要綁定到容器的 ID ,第二個(gè)參數(shù)是一個(gè)回調(diào)函數(shù)每當(dāng)執(zhí)行FileSessionStorage
類時(shí)執(zhí)行,我們還可以傳遞一個(gè)表示類名的字符串,如下所示。
Note: 如果你查看 Laravel 包時(shí),你將看到綁定有時(shí)會(huì)分組,比如(view
,view.finder
……)。
假設(shè)我們將會(huì)話存儲(chǔ)轉(zhuǎn)換為 Mysql 存儲(chǔ),我們的類應(yīng)該類似于:
class MysqlSessionStorage{ public function __construct(){ //... } public function get($key){ // do something } public function set( $key, $value ){ // do something } }
現(xiàn)在我們已經(jīng)更改了依賴項(xiàng),我們還需要更改SimpleAuth
構(gòu)造函數(shù),并將新對(duì)象綁定到容器中!
高級(jí)模塊不應(yīng)該依賴于低級(jí)模塊,兩者都應(yīng)該依賴于抽象對(duì)象。
抽象不應(yīng)該依賴于細(xì)節(jié),細(xì)節(jié)應(yīng)該取決于抽象。Robert C. Martin
我們的SimpleAuth
類不應(yīng)該關(guān)心我們的存儲(chǔ)是如何完成的,相反它更應(yīng)該關(guān)注于消費(fèi)的服務(wù)。
因此,我們可以抽象實(shí)現(xiàn)我們的存儲(chǔ):
interface SessionStorage{ public function get( $key ); public function set( $key, $value ); }
這樣我們就可以實(shí)現(xiàn)并請(qǐng)求SessionStorage
接口的實(shí)例:
class FileSessionStorage implements SessionStorage{ public function __construct(){ //... } public function get( $key ){ //... } public function set( $key, $value ){ //... } } class MysqlSessionStorage implements SessionStorage{ public function __construct(){ //... } public function get( $key ){ //... } public function set( $key, $value ){ //... } } class SimpleAuth{ protected $session; public function __construct( SessionStorage $session ){ $this->session = $session; } }
如果我們使用App::make('SimpleAuth')
通過容器解析SimpleAuth
類,容器將會(huì)拋出BindingResolutionException
,嘗試從綁定解析類之后,返回到反射方法并解析所有依賴項(xiàng)。
Uncaught exception 'Illuminate\Container\BindingResolutionException' with message 'Target [SessionStorage] is not instantiable.'
容器正試圖將接口實(shí)例化。我們可以為該接口做一個(gè)具體的綁定。
App:bind( 'SessionStorage', 'MysqlSessionStorage' );
現(xiàn)在每次我們嘗試從容器解析該接口時(shí),我們會(huì)得到一個(gè)MysqlSessionStorage
實(shí)例。如果我們想要切換我們的存儲(chǔ)服務(wù),我們只要變更一下這個(gè)綁定。
Note: 如果你想要查看一個(gè)類是否已經(jīng)在容器中被綁定,你可以使用App::bound('ClassName')
,或者可以使用App::bindIf('ClassName')
來注冊(cè)一個(gè)還未被注冊(cè)過的綁定。
Laravel Ioc 也提供App::singleton('ClassName', 'resolver')
來處理單例的綁定。
你也可以使用App::instance('ClassName', 'instance')
來創(chuàng)建單例的綁定。
如果容器不能解析依賴項(xiàng)就會(huì)拋出ReflectionException
,但是我們可以使用App::resolvingAny(Closure)
方法以回調(diào)函數(shù)的形式來解析任何指定的類型。
Note: 如果你為某個(gè)類型已經(jīng)注冊(cè)了一個(gè)解析方式resolvingAny
方法仍然會(huì)被調(diào)用,但它會(huì)直接返回bind
方法的返回值。
這些綁定寫在哪兒:
如果只是一個(gè)小型應(yīng)用你可以寫在一個(gè)全局的起始文件global/start.php
中,但如果項(xiàng)目變得越來越龐大就有必要使用 Service Provider 。
測(cè)試:
當(dāng)需要快速簡(jiǎn)易的測(cè)試可以考慮使用php artisan tinker
,它十分強(qiáng)大,且能幫你提升你的 Laravel 測(cè)試流程。
Reflection API:
PHP 的 Reflection API 是非常強(qiáng)大的,如果你想要深入 Laravel Ioc 你需要熟悉 Reflection API ,可以先看下這個(gè) 教程 來獲得更多的信息。
以上是“Laravel中的依賴注入和IoC的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!