程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> 深入理解PHP中的匿名函數

深入理解PHP中的匿名函數

編輯:PHP綜合
匿名函數的作用就是擴大函數的使用功能,在PHP 5.3以前,傳遞Callback的方式,我們只有倆種選擇:

  ◆字符串的函數名

  ◆使用create_function的返回

  在PHP5.3以後, 我們多了一個選擇, 也就是Closure。

 

  1. $func = function () { ... };   
  2. array_walk($arr, $func);  

  從實現上來說, 第一種方式: 傳遞函數名字符串是最簡單的。而第二種方式create_function, 其實和第一種方式本質上一樣的, create_function返回一個字符串的函數名, 這個函數名的格式是:

 

  1. "\000_lambda_" . count(anonymous_functions)++  

  我們來看看create_function的實現步驟:

  1. 獲取參數, 函數體;

  2. 拼湊一個"function __lambda_func (參數) { 函數體;} "的字符串;

  3. eval;

  4. 通過__lambda_func在函數表中找到eval後得到的函數體, 找不到就出錯;

  5. 定義一個函數名:"\000_lambda_" . count(anonymous_functions)++;

  6. 用新的函數名替換__lambda_func;

  7. 返回新的函數。

  我們來驗證下:

 

  1. <?PHP  
  2. create_function("", 'echo __FUNCTION__;');  
  3. call_user_func("\000lambda_1", 1);  
  4. ?>  
  5. //輸出  __lambda_fun  

  因為在eval的時候, 函數名是”__lambda_func”, 所以匿名函數內會輸出__lambda_func, 而因為最後用”\000_lambda_”.count(anonymous_functions)++重命名了函數表中的”__lambda_func”函數, 所以可通過”\000_lambda_” . count(anonymous_functions)++調用這個匿名函數。為了證實這一點, 可以將create_function的返回值dump出來查看。

  而在PHP 5.3發布的時候, 其中有一條new feature就是支持閉包/Lambda Function, 我第一反應是以為zval新增了一個IS_FUNCTION, 但實際上是構造了一個PHP 5.3引入的Closure”類”的實例, Closure類的構造函數是私有的, 所以不能被直接實例化, 另外Closure類是Final類, 所以也不能做為基類派生子類.

 

  1. //PHP-5.3.0  
  2. $class = new ReflectionClass("Closure");  
  3. var_dump($class->isInternal());  
  4. var_dump($class->isAbstract() );  
  5. var_dump($class->isFinal());  
  6. var_dump($class->isInterface());  
  7. //輸出:  
  8. bool(true)  
  9. bool(false)  
  10. bool(true)  
  11. bool(false)  
  12. ?>  

  而PHP 5.3中對閉包的支持, 也僅僅是把要保持的外部變量, 做為Closure對象的”Static屬性”(並不是普通意義上的可遍歷/訪問的屬性).

 

  1. //PHP-5.3.0  
  2. $b = "laruence";  
  3. $func = function($a) use($b) {};  
  4. var_dump($func);  
  5. /* 輸出:  
  6. object(Closure)#1 (2) {  
  7. ["static"]=>  
  8.  array(1) {  
  9. ["b"]=> string(8) "laruence"   
  10. }    
  11. ["parameter"]=>   
  12. array(1) {   
  13.  ["$a"]=> string(10) "<required>"   
  14.  }   
  15.  }   
  16.  */  

  這個實現, 個人認為和JS對閉包的支持比起來, 還是有些太簡陋了。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved