1. 程式人生 > >從匿名函數(閉包特性)到 PHP 設計模式之容器模式

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

pos code data- them 獲取 常用 per sdn 數據

匿名函數(匿名函數)

  匿名函數,也叫閉包函數,它允許臨時創建一個沒有指定名稱的函數,常用作回調函數參數的值,也可以作為變量的值來使用。具體的使用見以下示例代碼:
/* 示例一:聲明一個簡單匿名函數,並賦值給一個變量,通過變量名調用這個匿名函數 */
$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() 方法獲取依賴的時候才執行匿名函數,並將實例化對象返回,這樣實現了按需實例化,不用則不實例化,提高了程序的運行效率。

參考文章出處:   1、http://www.thinkphp.cn/topic/13624.html
  2、https://www.cnblogs.com/sweng/p/6430374.html   3、http://www.jb51.net/article/114723.htm
  4、http://blog.csdn.net/realghost/article/details/35212285
  5、《跟兄弟連學 PHP》一書中《PHP 匿名函數和閉包》章節以及相關筆記

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