PHP5.3閉包特性及應用詳解

09年7月發佈的PHP 5.3 版本帶來瞭很多新的特性,, 其中比較惹眼的特性之一就是支持瞭閉包;那麼以後,我們也可以和那幫寫 Ruby、Javascript 等等“高科技語言”的傢夥們一樣,寫出非常酷的代碼嗎?呃,其實大部分情況下是可以的,而有些方面還是令人非常的困擾,下面慢慢道來。

很多語言的都提供瞭非常優雅和漂亮的操作數組的方法。在下面的例子中,會使用PHP5.3閉包特性以及其他語言提供的閉包功能,用於展示如何“客觀的”操作迭代數組。

譯註:原文作者比較火星,我不瞭解 Groovy 以及 Scala語言,所以這裡我加上 Javascript 的實現。

在開始之前先說明下,本例子僅僅是闡明觀點,並沒有考慮性能等其他方面的因素。
 

“貨比三傢”

用個簡單的例子開始,有下面個數組:

$nums = array(10, 20, 30, 40);

需要找出數組中大於 15 的項。那麼,不考慮閉包的情況下,我們或許會這樣寫:

  1. $res = array();  
  2. foreach ($nums as $n) {   
  3.    if ($n > 15) {        $res[] = $n;      
  4. }  

如果語言本身有閉包支持的,那麼或許會這樣寫(Groovy 語言)

def res = nums.findAll { it > 15 }或者使用 Scala 語言

val res = nums filter (_ > 15)譯註:Javascript 1.6 的話會是如下

var res = nums.filter(function(c){return c > 15});

因為循環操作已被抽象起來,所以可以看到 Groovy 、Scala (以及 Javascript) 都很漂亮得用一行就可以搞定。

當然,如果使用 PHP5.3 的閉包,也可以做到

$res = array_filter($nums, function($v) { return $v > 15; });

PHP 在這方面使用瞭比 Scala 更多的字符,但對比先前的例子,它更簡短並且能更好得閱讀。

順便說下,上面的 PHP 代碼實際上是使用瞭 Lambda 解析式,並不是個真正的閉包,這個 並不是我們目前關註的重點。詳細闡述 PHP 閉包以及 Lambda 解析式的資料,可以參考這裡。

目前看來感覺都還不錯,那麼我們再的題目增加點難度:找到所有大於 15 的項, 然後乘以 2 再加上作用域中的的某個變量值以後再返回。

Groovy 的實現:

  1. def x = 1def   
  2. res = nums .findAll { it > 15 } .collect { it * 2 + x } 

Scala 的實現:

  1. val x =   
  2. 1val res = nums filter (_ > 15) map (_ * 2 + x) 

PHP的實現:

  1. $x = 1;  
  2. $res = array_map(  
  3.     function($v) use ($x) {   
  4.         return $v * 2 + $x; },      
  5.      array_filter(        $nums,          
  6. function($v) { return $v > 15; })  
  7. );  

光從代碼量方面,現在看起來 PHP 與其他語言有出入瞭。先拋開代碼字面上本身 的審美不談,上面的 PHP 代碼還有個額外的問題。

例如,如果需要使用數組的鍵而非值作比較,怎麼辦?是的,上面的代碼就辦不到瞭。同時,從語法角度上說,上面的代碼非常難以閱讀。

返璞歸真,這時還是得返回老土的思路去解決問題:

  1. $x = 1;  
  2. $res = array();  
  3. foreach ($nums as $n) {  
  4.     if ($n > 15) {  
  5.         $res[] = $n * 2 + $x;  
  6.     }  

這樣看起來又很清楚瞭。但這個時候你或許又會迷惑瞭:“那還瞎折騰啥,這不就是個數組操作嗎?”。

是的,好戲還在後頭。這個時候該讓 PHP 的某些高級特性出場,來搞定這看似有自殘傾向 的“無聊問題”。

ArrayObject – 對數組的封裝

PHP 有個稱作 SPL 的標準庫,其中包含瞭個叫做 ArrayObject 的類,它能提供“像數組一 樣操作類”的功能,例如

  1. $res = new ArrayObject(array(10, 20, 30, 40));  
  2. foreach ($res as $v) {  
  3.     echo "$vn";  

ArrayObject 是個內置的類,所以你可以像其他類類操作一樣封裝它。

Arr – 包上糖衣

既然我們已經有瞭 ArrayObject 以及閉包這些特性,我們就可以開始嘗試封裝它:

  1. class Arr extends ArrayObject{      
  2. static function make($array)    {  
  3.         return new&nb

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *