第一重境界
假設場景:我們需要寫一個處理類,能夠同時操作會話,數據庫和文件系統。我們或許會這麼寫。
境界特徵:可以運行,但是嚴重耦合
<?php
/**
* 第一重
*/
namespace test1;
class DB{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class FileSystem{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class Session{
public function __construct($arg1,$arg2) {
echo 'constructed'.PHP_EOL;
}
}
class Writer{
public function __construct()
{
$db=new DB(1,2);
$filesystem=new FileSystem(3,4);
$session=new Session(5,6);
}
}
$writer=new Writer();
寫法缺點:
1.在公有函數中構造對象,一旦涉及到如數據庫參數的變動,修改會有很大的工作量
2.負責設計Writer類的人員需要對DB等類的各種API要熟悉
有沒有辦法降低耦合度?
第二重境界(參數依賴)
假設場景:數據庫地址因爲客戶不同,需要經常更換,調用到DB的類很多(假如有幾十個),希望即使更改了數據庫地址,也不用去修改這些類的代碼。
<?php
/**
* 第二重
*/
namespace test2;
class DB{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class FileSystem{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class Session{
public function __construct($arg1,$arg2) {
echo 'constructed'.PHP_EOL;
}
}
class Writer{
protected $_db;
protected $_filesystem;
protected $_session;
public function Set($db,$filesystem,$session){
$this->_db=$db;
$this->_filesystem=$filesystem;
$this->_session=$session;
}
}
$db=new DB(1,2);
$filesystem=new FileSystem(3,4);
$session=new Session(5,6);
$writer=new Writer();
$writer->Set($db,$filesystem,$session);
雖然把DB類的構造移到了客戶端,一旦涉及修改,工作量大大降低,但是新問題來了:爲了創建一個Writer類,我們需要先創建好DB類,FileSystem類等,這對負責涉及Writer類的人來說,要求是很高的,他需要看很多其他類文檔,一個個創建(可能還需要初始化),然後才能創建出他要的writer變量。
所以,我們希望,能有一種更好的寫法,使得寫Writer類的人,用一種更加快捷的接口,就能創建和調用他要的類,甚至連參數都不用填。
第三重境界(IOC容器)
經過前兩重境界,我們希望能新增以下這些好處:
- 希望DB類,Session類,FileSystem類“拿來即用”,不用每次繁瑣的初始化,比如寫$db=new DB(arg1,arg2);這類語句。
- 希望DB等類型的對象是“全局”,在整個程序運行期間,隨時可以調用。
- 調用DB等類型的程序員不用知道這個類太多的細節,甚至可以用一個字符串的別名來創建這樣一個對象。
能夠實現以上目標的就是IOC容器,可以把IOC容器簡單的看成一個全局變量,並用關聯數組把字符串和構造函數做綁定。
<?php
/**
* 第三重
*/
namespace test3;
class DB{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class FileSystem{
public function __construct($arg1,$arg2){
echo 'constructed'.PHP_EOL;
}
}
class Session{
public function __construct($arg1,$arg2) {
echo 'constructed'.PHP_EOL;
}
}
class Container{
public $bindings;
public function bind($abstract,$concrete){
$this->bindings[$abstract]=$concrete;
}
public function make($abstract,$parameters=[]){
return call_user_func_array($this->bindings[$abstract],$parameters);
}
}
$container=new Container();
$container->bind('db',function ($arg1,$arg2){return new DB($arg1,$arg2);});
$container->bind('filesystem',function ($arg1,$arg2){return new FileSystem($arg1,$arg2);});
$container->bind('session',function ($arg1,$arg2){return new Session($arg1,$arg2);});
class Writer{
protected $_db;
protected $_filesystem;
protected $_session;
protected $container;
public function __construct(Container $container)
{
$this->_db=$container->make('db',[1,2]);
$this->_filesystem=$container->make('filesystem',[3,4]);
$this->_session=$container->make('session',[5,6]);
}
}
$writer=new Writer($container);