控制反轉和依賴注入
控制反轉(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