通俗易懂理解PHP依賴注入容器

2020-07-16 10:06:37

依賴注入容器理解

耦合

一個好的程式碼結構設計一定是松耦合的,這也是很多通用設計模式的宗旨,就是把分散在各處的同一個功能的程式碼匯聚到一起,形成一個模組,然後在不同模組之間通過一些細小的、明確的渠道進行溝通。

在實踐中,不同功能和模組之間的互相依賴是不可避免的,而如何處理好這些依賴之間的關係則是程式碼結構能否變得美好的關鍵。

<?php
class User
{
    public function register($user)
    {
        // 註冊操作
        ...
 
        // 傳送確認郵件
        $notify = new Notify();
        $notify->sendEmail('register', $user);
    }
}
 
class Notify
{
    public function sendEmail($type, $data)
    {
        switch $type {
            case 'register':
                // 傳送註冊確認郵件
                $email = new Email($type);
                $email->send($data);
            ...
        }
    }
}
 
class Email
{
    public function send($data)
    {
        // 傳送郵件
    }
}

上述程式碼中,三個類之間逐層依賴,三個類範例化的順序是 User -> Notify -> Email

也就是說我先範例化User類,可能執行了一些程式碼之後再去範例化我需要的其他類,比如Notify,以此類推。

這種依賴會讓我們不得不為了得到需要的依賴而去做的一些準備工作,有時候可能一個new操作還不夠。而這部分工作就是所說的耦合,他會讓一個獨立功能的類不得不去關心一些和自己的主體功能沒什麼關係的操作。

解除一個類對其他類的依賴

要解決這個問題也很簡單,我可以先範例化好Email類,然後再範例化Notify,然後把Email物件作為引數傳給Notify,最後範例化User類,然後把Notify傳進去。這就是所謂的依賴注入,可以看到這個過程中類範例化的順序完全反過來了,先範例化被依賴的物件,而不是先範例化最終需要的物件,這是控制反轉。

程式碼如下:

<?php
$email = new Email();
$notify = new Notify($email);
$user = new User($notify);

可以通過建構函式來注入需要的依賴,也可以用一些其他的方法。

用容器託管依賴

那又有新的問題,例子中只有三個類還好,那如果這個User類依賴Notify來發郵件,依賴Model來存資料庫,依賴redis來快取,這樣固然把依賴關係轉移到了類的外部,但還是會導致我只想範例化一下User的時候,卻要手動做很多的準備工作,會讓程式碼混亂。所以這個時候需要一個容器。而這個容器的作用就是替我來管理這些依賴。

<?php
// 容器
class Container implements ArrayAccess
{
    protected $values = [];
 
    public function offsetGet($offset) 
    {
        return $this->values[$offset]($this);
    }
 
    public function offsetSet($offset, $value) 
    {
        $this->values[$offset] = $value;
    }
}

在程式啟動的時候,我們可以在一個地方統一的註冊好一系列的基礎服務。

<?php
$container = new Container();
 
$container['notify'] = function($c) {
    return new Notify();
};
 
$container['email'] = function($c) {
    return new Email();
};

就會變成這樣

<?php
class User
{
    public function register($user)
    {
        // 註冊操作
        ...
 
        // 傳送確認郵件
        $container('notify')->sendEmail('register', $user);
    }
}
 
class Notify
{
    public function sendEmail($type, $data)
    {
        switch $type {
            case 'register':
                // 傳送註冊確認郵件
                $email = $container['email'];
                $email->send($data);
            ...
        }
    }
}
 
class Email
{
    public function send($data)
    {
        // 傳送郵件
    }
}

就是當User需要Notify的時候,會去向容器要這個類的物件,那至於Notify再依賴什麼其他的東西,我就不用管了,因為Notify也會去向容器要它自己需要的依賴。所有這些依賴關係的處理就完全託管給容器了,我們既不需要去關心依賴之間的層次關係,也避免了依賴之間的耦合。

需要注意的是,依賴注入容器一般只接受一個匿名函數,而不是一個範例化好的物件,匿名函數會告訴容器怎樣去獲得一個物件,這樣可以使得一個服務在被需要的時候才會去範例化

以上就是通俗易懂理解PHP依賴注入容器的詳細內容,更多請關注TW511.COM其它相關文章!