PHP中怎么使用回調(diào)函數(shù)與匿名函數(shù),相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。
成都創(chuàng)新互聯(lián)公司專注于光澤網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗(yàn)。 熱誠(chéng)為您提供光澤營(yíng)銷型網(wǎng)站建設(shè),光澤網(wǎng)站制作、光澤網(wǎng)頁(yè)設(shè)計(jì)、光澤網(wǎng)站官網(wǎng)定制、微信小程序開發(fā)服務(wù),打造光澤網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供光澤網(wǎng)站排名全網(wǎng)營(yíng)銷落地服務(wù)。回調(diào)函數(shù)和匿名函數(shù)
回調(diào)函數(shù)、閉包在JS中并不陌生,JS使用它可以完成事件機(jī)制,進(jìn)行許多復(fù)雜的操作。PHP中卻不常使用,今天來說一說PHP中中的回調(diào)函數(shù)和匿名函數(shù)。
回調(diào)函數(shù)
回調(diào)函數(shù):Callback (即call then back 被主函數(shù)調(diào)用運(yùn)算后會(huì)返回主函數(shù)),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。
通俗的解釋就是把函數(shù)作為參數(shù)傳入進(jìn)另一個(gè)函數(shù)中使用;PHP中有許多 “需求參數(shù)為函數(shù)” 的函數(shù),像array_map,usort,call_user_func_array之類,他們執(zhí)行傳入的函數(shù),然后直接將結(jié)果返回主函數(shù)。好處是函數(shù)作為值使用起來方便,而且代碼簡(jiǎn)潔,可讀性強(qiáng)。
匿名函數(shù)
匿名函數(shù),顧名思義,是沒有一個(gè)確定函數(shù)名的函數(shù),PHP將匿名函數(shù)和閉包視作相同的概念(匿名函數(shù)在PHP中也叫作閉包函數(shù))。它的用法,當(dāng)然只能被當(dāng)作變量來使用了。
PHP中將一個(gè)函數(shù)賦值給一個(gè)變量的方式有四種:
① 我們經(jīng)常會(huì)用到的:函數(shù)在外部定義/或PHP內(nèi)置,直接將函數(shù)名作為字符串參數(shù)傳入。注意:如果是類靜態(tài)函數(shù)的話以CLASS::FUNC_NAME
的方式傳入。
② 使用create_function($args, $func_code);
創(chuàng)建函數(shù),會(huì)返回一個(gè)函數(shù)名。 $func_code為代碼體,$args為參數(shù)字符串,以','分隔;
③ 直接賦值:$func_name = function($arg){statement}
④ 直接使用匿名函數(shù),在參數(shù)處直接定義函數(shù),不賦給具體的變量值;
第一種方式因?yàn)槭瞧匠K茫辉俣嗵?;第二種類似eval()方法的用法,也被PHP官方列為不推薦使用的方式,而且其定義方式太不直觀,我除了測(cè)試外,也沒有在其他地方使用過,也略過不提。在這里重點(diǎn)說一下第三種和第四種用法;
后兩種創(chuàng)建的函數(shù)就被稱為匿名函數(shù),也就是閉包函數(shù), 第三種賦值法方式創(chuàng)建的函數(shù)非常靈活,可以通過變量引用。可以用 is_callable($func_name) 來測(cè)試此函數(shù)是否可以被調(diào)用, 也可以通過$func_name($var)來直接調(diào)用;而第四種方式創(chuàng)建的函數(shù)比較類似于JS中的回調(diào)函數(shù),不需要變量賦值,直接使用;
另外要特別介紹的是 use 關(guān)鍵詞,它可以在定義函數(shù)時(shí),用來引用父作用域中的變量;用法為 function($arg) use($outside_arg) {function_statement} 。其中$outside_arg 為父作用域中的變量,可以在function_statement使用。
這種用法用在回調(diào)函數(shù)“參數(shù)值數(shù)量確定”的函數(shù)中。 如usort需求$callback的參數(shù)值為兩項(xiàng),可是我們需要引入別的參數(shù)來影響排序怎么辦呢?使用use()關(guān)鍵詞就很方便地把一個(gè)新的變量引入$callback內(nèi)部使用了。
array_map/array_filter/array_walk:
把這三個(gè)函數(shù)放在一塊是因?yàn)檫@三個(gè)函數(shù)在執(zhí)行邏輯上比較類似,類似于下面的代碼:
$result = []; foreach($vars as $key=>$val){ $item = callback(); $result[] = $item; } return $result; array_walk($vars, $callback)
其callback應(yīng)如下:
$callback = function(&$val, $key[, $arg]){ doSomething($val); }
array_walk返回執(zhí)行是否成功,是一個(gè)布爾值。對(duì)$value添加引用符號(hào)可以在函數(shù)內(nèi)改變$value值,以達(dá)到改變$vars數(shù)組的效果。由于其$callback對(duì)參數(shù)數(shù)量要求為兩項(xiàng),array_walk不能傳入strtolower/array_filter之類的$callback,若想實(shí)現(xiàn)類似功能,可以使用接下來要說的array_map()
。
array_walk_recursive($arr, $callback);
返回值和執(zhí)行機(jī)制類似于array_walk;
其callback同array_walk,不同的是,如果$val是數(shù)組,函數(shù)會(huì)遞歸地向下處理$val;需要注意的是這樣的話$val為數(shù)組的$key就會(huì)被忽略掉了。
array_filter($vars, $callback, $flag);
其$callback類似于:
$callback = function($var){ return true or false; }
array_filter會(huì)過濾掉$callback執(zhí)行時(shí)返回為false的項(xiàng)目,array_filter返回過濾完成后的數(shù)組。
第三個(gè)參數(shù) $flag決定其callback形參$var的值,不過這個(gè)可能是PHP高版本的特性,我的PHP5.5.3不支持,大家可以自行測(cè)試。默認(rèn)傳入數(shù)組每項(xiàng)的value,當(dāng)flag為ARRAY_FILTER_USE_KEY傳入數(shù)組每項(xiàng)的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經(jīng)過callback處理后的數(shù)組(會(huì)改變?cè)瓟?shù)組);如果有多個(gè)數(shù)組的時(shí)候?qū)蓚€(gè)數(shù)組同樣順序的項(xiàng)目傳入處理,執(zhí)行次數(shù)為參數(shù)數(shù)組中項(xiàng)目最多的個(gè)數(shù);
usort/array_reduce
把這兩個(gè)函數(shù)放在一塊,因?yàn)樗麄兊膱?zhí)行機(jī)制都有些特殊。
usort(&$vars, $callback)
$callback應(yīng)該如下:
callback = function($left, $right){ $res = compare($left, $right); return $res; }
usort返回執(zhí)行成功與否,bool值。用戶自定義方法 比較$left 和 $right,其中$left和$right是$vars中的任意兩項(xiàng);
$left > $right時(shí)返回 正整數(shù), $left < $right時(shí)返回 負(fù)整數(shù), $left = $right時(shí)返回0;
$vars中的元素會(huì)被取出會(huì)被由小到大升序排序。 想實(shí)現(xiàn)降序排列,將$callback的返回值反一下就行了。
array_reduce($vars ,$callable [, mixed $initial = NULL])
$callback應(yīng)該如下:
$callback = function($initial, $var){ $initial = calculate($initail, $var); return $initial; }
初始值$initial默認(rèn)為null,返回經(jīng)過迭代后的initial;一定要將$initial返回,這樣才能不停地改變$initial的值,實(shí)現(xiàn)迭代的效果。
這里順便說一下map和reduce的不同:
map:將數(shù)組中的成員遍歷處理,每次返回處理后的一個(gè)值,最后結(jié)果值為所有處理后值組成的多項(xiàng)數(shù)組;
reduce:遍歷數(shù)組成員,每次使用數(shù)組成員結(jié)合初始值處理,并將初始值返回,即使用上一次執(zhí)行的結(jié)果,配合下一次的輸入繼續(xù)產(chǎn)生結(jié)果,結(jié)果值為一項(xiàng);
call_user_func/call_user_func_array
call_user_func[_array]($callback, $param)
$callback形如:
$callback = function($param){ $result = statement(); return $result; }
返回值多種,具體看$callback。
可用此函數(shù)實(shí)現(xiàn)PHP的事件機(jī)制,其實(shí)并不高深,在判斷條件達(dá)成,或程序執(zhí)行到某一步后 call_user_func()就OK了。這個(gè)我在之前的博客中也有介紹到:搭建自己的PHP框架
總結(jié)
其實(shí)以上$callback不用單獨(dú)定義并使用變量引用,使用上面說過的第四種函數(shù)定義方式,直接在函數(shù)內(nèi)定義,使用‘完全'匿名函數(shù)就行了。 如:
usort($records, function mySortFunc($arg) use ($order){ func_statement; });
看完上述內(nèi)容,你們掌握PHP中怎么使用回調(diào)函數(shù)與匿名函數(shù)的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!