首頁 > 後端開發 > php教程 > PHP回呼函數的解析

PHP回呼函數的解析

*文
發布: 2023-03-18 08:52:02
原創
2799 人瀏覽過

熟悉node開發的人對回呼函數一定不陌生,但在PHP中卻不常用。其實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的闭包(Closure)匿名函数初探

详解PHP匿名函数与注意事项

PHP匿名函数和use子句用法实例,匿名use子句实例_PHP教程

以上是PHP回呼函數的解析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板