溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Laravel中的依賴注入和IoC的示例分析

發(fā)布時間:2021-01-25 12:44:08 來源:億速云 閱讀:163 作者:小新 欄目:編程語言

這篇文章主要介紹Laravel中的依賴注入和IoC的示例分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

依賴注入

依賴注入一詞是由 Martin Fowler 提出的術(shù)語,它是將組件注入到應(yīng)用程序中的一種行為。就像 Ward Cunningham 說的:

依賴注入是敏捷架構(gòu)中關(guān)鍵元素。

讓我們看一個例子:

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;
    }
}

如果你要測試或者維護這個類,你必須訪問數(shù)據(jù)庫的實例來進行一些查詢。為了避免必須這樣做,你可以將此類與其他類進行 解耦 ,你有三個選項之一,可以將 Connection 類注入而不需要直接使用它。

將組件注入類時,可以使用以下三個選項之一:

構(gòu)造方法注入

class UserProvider{
    protected $connection;

    public function __construct( Connection $con ){
        $this->connection = $con;
    }
    ...

Setter 方法注入

同樣,我們也可以使用 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)一個類實現(xiàn)了我們的接口時,我們定義了 injectConnection 方法來解決依賴關(guān)系。

優(yōu)勢

現(xiàn)在,當(dāng)測試我們的類時,我們可以模擬依賴類并將其作為參數(shù)傳遞。每個類必須專注于一個特定的任務(wù),而不應(yīng)該關(guān)心解決它們的依賴性。這樣,你將擁有一個更專注和可維護的應(yīng)用程序。

如果你想了解更多關(guān)于 DI 的信息,Alejandro Gervassio 在 本系列 文章中對其進行了廣泛而專業(yè)的介紹,所以一定要去讀這些文章。那么,什么又是 IoC 呢?IoC (控制反轉(zhuǎn))不需要使用依賴注入,但它可以幫助你有效的管理依賴關(guān)系。

控制反轉(zhuǎn)

Ioc 是一個簡單的組件,可以更加方便地解析依賴項。你可以將對象形容為容器,并且每次解析類時,都會自動注入依賴項。

Laravel Ioc

當(dāng)你請求一個對象時, Laravel Ioc 在解決依賴關(guān)系的方式上有些特殊:

Laravel中的依賴注入和IoC的示例分析

我們使用一個簡單的例子,將在本文中改進它。
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)建一個 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)建一個對象:

$auth = new SimpleAuth( new FileSessionStorage() );

現(xiàn)在我想使用 Laravel Ioc 來管理這一切。

因為 Application 類繼承自 Container 類,所以你可以通過 App 門面來訪問容器。

App::bind( 'FileSessionStorage', function(){
    return new FileSessionStorage;
});

bind 方法第一個參數(shù)是要綁定到容器的唯一 ID ,第二個參數(shù)是一個回調(diào)函數(shù)每當(dāng)執(zhí)行 FileSessionStorage 類時執(zhí)行,我們還可以傳遞一個表示類名的字符串,如下所示。

Note: 如果你查看 Laravel 包時,你將看到綁定有時會分組,比如( view, view.finder……)。

假設(shè)我們將會話存儲轉(zhuǎn)換為 Mysql 存儲,我們的類應(yīng)該類似于:

class MysqlSessionStorage{

  public function __construct(){
    //...
  }

  public function get($key){
    // do something
  }

  public function set( $key, $value ){
    // do something
  }
}

現(xiàn)在我們已經(jīng)更改了依賴項,我們還需要更改 SimpleAuth 構(gòu)造函數(shù),并將新對象綁定到容器中!

高級模塊不應(yīng)該依賴于低級模塊,兩者都應(yīng)該依賴于抽象對象。
抽象不應(yīng)該依賴于細節(jié),細節(jié)應(yīng)該取決于抽象。

Robert C. Martin

我們的 SimpleAuth 類不應(yīng)該關(guān)心我們的存儲是如何完成的,相反它更應(yīng)該關(guān)注于消費的服務(wù)。

因此,我們可以抽象實現(xiàn)我們的存儲:

interface SessionStorage{
  public function get( $key );
  public function set( $key, $value );
}

這樣我們就可以實現(xiàn)并請求 SessionStorage 接口的實例:

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
類,容器將會拋出 BindingResolutionException ,嘗試從綁定解析類之后,返回到反射方法并解析所有依賴項。

Uncaught exception 'Illuminate\Container\BindingResolutionException' with message 'Target [SessionStorage] is not instantiable.'

容器正試圖將接口實例化。我們可以為該接口做一個具體的綁定。

App:bind( 'SessionStorage', 'MysqlSessionStorage' );

現(xiàn)在每次我們嘗試從容器解析該接口時,我們會得到一個 MysqlSessionStorage 實例。如果我們想要切換我們的存儲服務(wù),我們只要變更一下這個綁定。

Note: 如果你想要查看一個類是否已經(jīng)在容器中被綁定,你可以使用 App::bound('ClassName') ,或者可以使用 App::bindIf('ClassName') 來注冊一個還未被注冊過的綁定。

Laravel Ioc 也提供 App::singleton('ClassName', 'resolver') 來處理單例的綁定。
你也可以使用 App::instance('ClassName', 'instance') 來創(chuàng)建單例的綁定。
如果容器不能解析依賴項就會拋出 ReflectionException ,但是我們可以使用 App::resolvingAny(Closure) 方法以回調(diào)函數(shù)的形式來解析任何指定的類型。

Note: 如果你為某個類型已經(jīng)注冊了一個解析方式 resolvingAny 方法仍然會被調(diào)用,但它會直接返回 bind 方法的返回值。

小貼士

  • 這些綁定寫在哪兒:
    如果只是一個小型應(yīng)用你可以寫在一個全局的起始文件 global/start.php 中,但如果項目變得越來越龐大就有必要使用 Service Provider 。

  • 測試:
    當(dāng)需要快速簡易的測試可以考慮使用 php artisan tinker ,它十分強大,且能幫你提升你的 Laravel 測試流程。

  • Reflection API:
    PHP 的 Reflection API 是非常強大的,如果你想要深入 Laravel Ioc 你需要熟悉 Reflection API ,可以先看下這個 教程 來獲得更多的信息。

以上是“Laravel中的依賴注入和IoC的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI