PHP回呼函數與匿名函數詳細解讀

來源:互聯網
上載者:User
這篇文章主要介紹了PHP回呼函數與匿名函數,結合執行個體形式分析了php回呼函數與匿名函數的具體功能、用法及相關注意事項,需要的朋友可以參考下

具體如下:

回呼函數和匿名函數

回呼函數、閉包在JS中並不陌生,JS使用它可以完成事件機制,進行許多複雜的操作。PHP中卻不常使用,今天來說一說PHP中中的回呼函數和匿名函數。

回呼函數

回呼函數:Callback (即call then back 被主函數調用運算後會返回主函數),是指通過函數參數傳遞到其它代碼的,某一塊可執行代碼的引用。

通俗的解釋就是把函數作為參數傳入進另一個函數中使用;PHP中有許多 “需求參數為函數” 的函數,像array_map,usort,call_user_func_array之類,他們執行傳入的函數,然後直接將結果返回主函數。好處是函數作為值使用起來方便,而且代碼簡潔,可讀性強。

匿名函數

匿名函數,顧名思義,是沒有一個確定函數名的函數,PHP將匿名函數和閉包視作相同的概念(匿名函數在PHP中也叫作閉包函數)。它的用法,當然只能被當作變數來使用了。

PHP中將一個函數賦值給一個變數的方式有四種:

① 我們經常會用到的:函數在外部定義/或PHP內建,直接將函數名作為字串參數傳入。注意:如果是類靜態函數的話以CLASS::FUNC_NAME的方式傳入。

② 使用create_function($args, $func_code);建立函數,會返回一個函數名。 $func_code為代碼體,$args為參數字串,以','分隔;

③ 直接賦值:$func_name = function($arg){statement}

④ 直接使用匿名函數,在參數處直接定義函數,不賦給具體的變數值;

第一種方式因為是平常所用,不再多提;第二種類似eval()方法的用法,也被PHP官方列為不推薦使用的方式,而且其定義方式太不直觀,我除了測試外,也沒有在其他地方使用過,也略過不提。在這裡重點說一下第三種和第四種用法;

後兩種建立的函數就被稱為匿名函數,也就是閉包函數, 第三種賦值法方式建立的函數非常靈活,可以通過變數引用。可以用 is_callable($func_name) 來測試此函數是否可以被調用, 也可以通過$func_name($var)來直接調用;而第四種方式建立的函數比較類似於JS中的回呼函數,不需要變數賦值,直接使用;

另外要特別介紹的是 use 關鍵詞,它可以在定義函數時,用來引用父範圍中的變數;用法為 function($arg) use($outside_arg) {function_statement} 。其中$outside_arg 為父範圍中的變數,可以在function_statement使用。

這種用法用在回呼函數“參數值數量確定”的函數中。 如usort需求$callback的參數值為兩項,可是我們需要引入別的參數來影響排序怎麼辦呢?使用use()關鍵詞就很方便地把一個新的變數引入$callback內部使用了。

array_map/array_filter/array_walk:

把這三個函數放在一塊是因為這三個函數在執行邏輯上比較類似,類似於下面的代碼:

$result = [];foreach($vars as $key=>$val){  $item = callback();  $result[] = $item;}return $result;array_walk($vars, $callback)

其callback應如下:

$callback = function(&$val, $key[, $arg]){    doSomething($val);}

array_walk返回執行是否成功,是一個布爾值。對$value添加引用符號可以在函數內改變$value值,以達到改變$vars數組的效果。由於其$callback對參數數量要求為兩項,array_walk不能傳入strtolower/array_filter之類的$callback,若想實作類別似功能,可以使用接下來要說的array_map()

array_walk_recursive($arr, $callback);

傳回值和執行機制類似於array_walk;

其callback同array_walk,不同的是,如果$val是數組,函數會遞迴地向下處理$val;需要注意的是這樣的話$val為數組的$key就會被忽略掉了。

array_filter($vars, $callback, $flag);

其$callback類似於:

$callback = function($var){  return true or false;     }

array_filter會過濾掉$callback執行時返回為false的項目,array_filter返回過濾完成後的數組。

第三個參數 $flag決定其callback形參$var的值,不過這個可能是PHP高版本的特性,我的PHP5.5.3不支援,大家可以自行測試。預設傳入數組每項的value,當flag為ARRAY_FILTER_USE_KEY傳入數組每項的key,ARRAY_FILTER_USE_BOTH傳入鍵和值;

array_map($callback, &$var_as [,$var_bs...]);

其$callback類似於:

$callback = function($var_a[, $var_b...]){  doSomething($var_a, $var_b);}

返回$var_as經過callback處理後的數組(會改變原數組);如果有多個數組的時候將兩個數組同樣順序的項目傳入處理,執行次數為參數數組中項目最多的個數;

usort/array_reduce

把這兩個函數放在一塊,因為他們的執行機制都有些特殊。

usort(&$vars, $callback)

$callback應該如下:

callback = function($left, $right){    $res = compare($left, $right);    return $res;}

usort返回執行成功與否,bool值。使用者自訂方法 比較$left 和 $right,其中$left和$right是$vars中的任意兩項;

$left > $right時返回 正整數, $left < $right時返回 負整數, $left = $right時返回0;

$vars中的元素會被取出會被由小到大升序排序。 想實現降序排列,將$callback的傳回值反一下就行了。

array_reduce($vars ,$callable [, mixed $initial = NULL])

$callback應該如下:

$callback = function($initial, $var){    $initial = calculate($initail, $var);    return $initial;}

初始值$initial預設為null,返回經過迭代後的initial;一定要將$initial返回,這樣才能不停地改變$initial的值,實現迭代的效果。

這裡順便說一下map和reduce的不同:

map:將數組中的成員遍曆處理,每次返回處理後的一個值,最後結果值為所有處理後值組成的多項數組;

reduce:遍曆數群組成員,每次使用數群組成員結合初始值處理,並將初始值返回,即使用上一次執行的結果,配合下一次的輸入繼續產生結果,結果值為一項;

call_user_func/call_user_func_array

call_user_func[_array]($callback, $param)

$callback形如:

$callback = function($param){    $result = statement();     return $result;}

傳回值多種,具體看$callback。

可用此函數實現PHP的事件機制,其實並不高深,在判斷條件達成,或程式執行到某一步後 call_user_func()就OK了。這個我在之前的部落格中也有介紹到:搭建自己的PHP架構

總結

其實以上$callback不用單獨定義並使用變數引用,使用上面說過的第四種函數定義方式,直接在函數內定義,使用‘完全'匿名函數就行了。 如:

usort($records, function mySortFunc($arg) use ($order){  func_statement;});

相關推薦:

PHP回呼函數與匿名函數使用案例解析

結合代碼詳細為你講解,php中的array_map,array_walk以及匿名函數

php中的匿名函數和閉包(closure)用法

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.