1. 程式人生 > 實用技巧 >控制反轉和依賴注入

控制反轉和依賴注入

控制反轉(IOC)

首先,我們來看一個例子。

class Person
{
   private $name = '';
   private $age = 0;

   public function __construct(string $name, int $age)
{
       $this->name = $name;
       $this->age = $age;
   }

   public function eat ()
{
       echo '吃東西' . PHP_EOL;
   }

   public function drink ()
{
       
echo '喝水' . PHP_EOL; } public function sleep () { echo '睡覺' . PHP_EOL; } public function wakeup () { echo '起床' . PHP_EOL; } public function drive () { echo '開車' . PHP_EOL; } public function wash () { echo '洗漱' . PHP_EOL; } }

小明現在早上起來需要去上班,那麼小明需要做以下事情

$person = new Person('小明', 24);
$person->wakeup();
$person->wash();
$person->eat();
echo '帶上車鑰匙、手機、電腦' .PHP_EOL;
$person->drive();

上面的流程都是由程式設計師自己控制的。現在,我們想辦法,讓框架來控制流程。我們在Person類中新增一個方法,程式碼如下:

public function work (callable $bring)
{
    $this->wakeup();
    $this->wash();
    
$this->eat(); $bring(); $this->drive(); }

小黃也需要去上班,現在他只安裝框架的指導就可以完成上班的動作了。

$person = new Person('小黃', 29);
$person->work(function ()
{
    echo '帶上手機、車鑰匙、檔案' . PHP_EOL;
});

修改後的程式碼完成了控制反轉,以前的程式碼整個上班的流程由程式設計師控制,修改後的是由框架控制上班的流程的。程式的流程控制由程式設計師“反轉”到了框架。

現在可以給出控制反轉的定義了:

實際上,控制反轉是一個比較籠統的設計思想,並不是一種具體的實現方法,一般用來指導框架層面的設計。這裡所說的“控制”指的是對程式執行流程的控制,而“反轉”指的是在沒有使用框架之前,程式設計師自己控制整個程式的執行。在使用框架之後,整個程式的執行流程通過框架來控制。流程的控制權從程式設計師“反轉”給了框架。

依賴注入

控制反轉是一種設計思想,而依賴注入是一種具體的編碼技巧,依賴注入是實現控制反轉最常用的技巧。依賴注入看起來“高大上”,實際上非常容易理解和掌握。

那到底什麼是依賴注入呢?我們用一句話來概括就是:不通過 new() 的方式在類內部建立依賴類物件,而是將依賴的類物件在外部建立好之後,通過建構函式、函式引數等方式傳遞(或注入)給類使用。

下面來看一個例項:

interface Log
{
   function write (string $msg);
}

class TextLog implements Log
{
   public function __construct($dirname, $txtname)
{
       $this->makeDir($dirname);
       $this->mkTxt($txtname);
   }

   private function makeDir (string $dirName) :void
{
       // do something
   }

   private function mkTxt (string $txtName) :void
{
       // do something
   }

   public function write (string $msg)
{
       // do something
   }
}

class RedisLog implements Log
{
   private $redis = null;
   private $key = '';

   public function __construct(string $key)
{
       $this->redis = '...'; // 獲取redis例項
       $this->key = $key;
       // ...
   }

   public function write (string $msg)
{
       // do something
   }
}

class App
{
   public function run ()
{
       // do something

       // 記錄日誌
       (new RedisLog('log'))->write('框架執行資訊記錄');
   }
}

可以看到,App類依賴RedisLog類,如果我們今後不再使用redis來記錄日子,而改用文字檔案的話,那麼就需要修改run裡面的程式碼。

現在,我們換成使用依賴注入這種技巧來改寫,程式碼如下;

class App
{
   private $logHandle = null;

   public function __construct(Log $log)
{
       $this->logHandle = $log;
   }

   public function run ()
{
       // do something

       // 記錄日誌
       $this->logHandle->write('框架執行資訊記錄');
   }
}

改寫後的App類不再依賴RedisLog類,可以隨時換成其他的Log類,只要該類實現了write方法即可。可以看到,使用了依賴注入,可以靈活的替換掉所依賴的類,另外它是編寫可測試程式碼最有效的手段。

連結:https://mp.weixin.qq.com/s/OhF8aPIFdOufGHXucg1uaQ