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。
第三個參數 $flag決定其callback形參$var的值,不過這個可能是PHP高版本的特性,我的PHP5.5.3不支持,大家可以自行測試。預設傳入數組每個項目的value,當flag為ARRAY_FILTER_USE_KEY傳入數組每個的value,當flag為ARRAY_FILTER_USE_KEY傳入數組每項的key,ARRAY_FILTER_USE_BOTH傳入鍵和值;
array_map($callback, &$var_)... ;
其$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; });
是不是逼格满满呢?
OK,介绍了几个用法~希望对大家有帮助,如果有问题,欢迎指出,如果您喜欢,可以点下推荐~

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

PHP和Python各有優勢,選擇依據項目需求。 1.PHP適合web開發,尤其快速開發和維護網站。 2.Python適用於數據科學、機器學習和人工智能,語法簡潔,適合初學者。
