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