從匿名函數(閉包特性)到 PHP 設計模式之容器模式

匿名函數(匿名函數)

  匿名函數,也叫閉包函數,它容許臨時建立一個沒有指定名稱的函數,經常使用做回調函數參數的值,也能夠做爲變量的值來使用。具體的使用見如下示例代碼:
/* 示例一:聲明一個簡單匿名函數,並賦值給一個變量,經過變量名調用這個匿名函數 */
$anonFunc = function($param){    
    echo $param;
}; 
$anonFunc('這裏是一個匿名函數');  // 經過變量名調用匿名函數,和普通函數沒什麼區別
/* 示例二:經過在函數內部使用匿名函數動態建立函數 */ function operate($operator){ if($operator == '+'){ return function($a, $b){ return $a + $b; } } if($operator == '-'){ return function($a, $b){ return $a - $b; } } } $add = operate('+'); echo $add(4, 3); // 7 $sub = operate('-'); echo $sub(4, 3); // 1 /* 示例三:匿名函數做爲回調函數參數傳入 */ function callback($callback){ $callback(); } function callback(){ // 閉包測試函數 echo '這裏是閉包測試函數體'; }
  以上代碼中的三個示例中,匿名函數都沒有進行傳參,咱們知道 在 JavaScript 中匿名函數用得很頻繁,並且父函數中的參數變量在子函數中能夠直接使用,可是 PHP 語言不容許這樣作,須要用到 use ($var) 關鍵字(注意代碼中的使用方式)實現一樣的目的。針對上面代碼中的示例三作以下修改:
/* 示例三修改:匿名函數做爲參數傳入,而且攜帶參數 */
function callback($callback) use ($content){
    $callback($content);
}
$content = '這裏是閉包函數的輸出內容';
function callback($content){
    // 閉包函數
    echo $content;
}
  以上代碼中的示例二,也能夠 經過 use 關鍵字實現匿名函數對父函數外層變量的引用。這些示例代碼中匿名函數和閉包特性的運用,只是爲了理解概念,並無多大的實戰意義,閉包的用途有不少,常見的是用在 PHP 框架中容器模式的依賴注入(DI)中。

PHP 面向對象之容器模式

    顧名思義,容器就是用來存放東西的,其實就是聲明一個類,專門用來存取對象實例,既然如此,那麼 容器裏至少要有兩個核心方法,以實現綁定依賴到容器和從容器獲取依賴。容器能夠說是一個依賴管理工具,有時候也叫作服務容器。
/* 聲明一個簡單的容器類 */
class Container{
    private $_diList = array();    // 用於存放依賴
/* 核心方法之一,用於綁定服務
    * @param string $className 類名稱
    * @param mixed $concrete 依賴在容器中的存儲方式,能夠是類名字符串,數組,一個實例化對象,或者是一個匿名函數
    */
    puclic function set($className, $concrete){
​
            $this->_diList[$className] = $concrete;   
    }
​
    /* 
    * 核心方法之二,用於獲取服務對象 
    * @param string $className 將要獲取的依賴的名稱
    * @return object 返回一個依賴的實例化對象
    */
    public function get($className){
        if(isset($this->_diList[$className])){
            return $this->diList[$className];
        }    
        return null;
    }
}
  以上代碼就是一個簡單的容器模式,其中的 set 方法用於註冊依賴,get 方法用於獲取依賴。 容器存儲依賴的方式有不少(具體參照筆記《PHP 面向對象之容器模式的依賴注入(DI)與控制反轉(Ioc)》),如下示例代碼以匿名函數的方式做爲說明。
/* 數據庫鏈接類 */
class Connection{
    public function __construct($dbParams){
        // connect the database...    
    }
    public someDbTask(){
        // code...
    }
}
/* 會話控制類 */
class Session{
    public function openSession(){
        session_start();
    }
    // code...
}
$container->set('session', function(){
    return new Session();
});
​
$container = new Container();
// 使用容器註冊數據庫鏈接服務
$container->set('db', function(){
    return new Connetion(array(  
        "host" => "localhost",  
        "username" => "root",  
        "password" => "root",  
        "dbname" => "dbname"  
    ));
});
// 使用容器註冊會話控制服務
$container->set('session', function(){
    return new Session();
});
// 獲取以前註冊到容器中的服務,並進行業務的處理
$container->get('db')->someDbTask();
$container->get('session')->openSession();

  以上代碼是對容器的使用方法,其中註冊了 db 和 session 兩個服務,這裏使用匿名函數做爲依賴的存儲方式,在調用 $container->set() 方法進行註冊服務時實際上並無進行實例化,而是在調用 $container->get() 方法獲取依賴的時候才執行匿名函數,並將實例化對象返回,這樣實現了按需實例化,不用則不實例化,提升了程序的運行效率。php

 
參考文章出處:
  五、 《跟兄弟連學 PHP》一書中《PHP 匿名函數和閉包》章節以及相關筆記
相關文章
相關標籤/搜索