PHP中的回調函數和匿名函數

来源:http://www.cnblogs.com/zhenbianshu/archive/2016/11/14/6063340.html
-Advertisement-
Play Games

回調函數、閉包在JS中並不陌生,JS使用它可以完成事件機制,進行許多複雜的操作。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;
});

是不是逼格滿滿呢?

OK,介紹了幾個用法~希望對大家有幫助,如果有問題,歡迎指出,如果您喜歡,可以點下推薦~

博客持續更新,歡迎大家關註。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C# 知識回顧 - 事件入門 【博主】反骨仔 【原文】http://www.cnblogs.com/liqingwen/p/6057301.html 序 之前通過《C# 知識回顧 - 委托 delegate》、《C# 知識回顧 - 委托 delegate (續)》介紹了委托的基本知識,這次我們來看看 ...
  • ASP.NET Core管道雖然在結構組成上顯得非常簡單,但是在具體實現上卻涉及到太多的對象,所以我們在 “通過重建Hosting系統理解HTTP請求在ASP.NET Core管道中的處理流程”(上篇、中篇、下篇) 中圍繞著一個經過極度簡化的模擬管道講述了真實管道構建的方式以及處理HTTP請求的流程... ...
  • 問題 我們想快速啟動一個 ASP.NET Web API 解決方案。 解決方案 APS.NET 模板一開始就支持 ASP.NET Web API。使用模板往我們的項目中添加 Controller,在我們解決方案的 Controllers 文件夾上右鍵,選擇“添加”->"Scaffolding"。 即 ...
  • 發現問題 需求很簡單,大致就是要批量往資料庫寫數據,於是打算用Parallel並行的方式寫入,希望能利用電腦多核特性加快程式執行速度。想的很美好,於是快速擼了類似下麵的一串代碼: 可意外的是竟然無情的報錯了: 奇葩的是當我再次刷新的時候異常又不一樣了,於是連著刷新好多次,總結出現過的異常有下麵這些 ...
  • 1.C#跨平臺物聯網通訊框架ServerSuperIO(SSIO)介紹 《連載 | 物聯網框架ServerSuperIO教程》1.4種通訊模式機制。 《連載 | 物聯網框架ServerSuperIO教程》2.服務實例的配置參數說明 《連載 | 物聯網框架ServerSuperIO教程》- 3.設備驅 ...
  • 黃金分割比例是自然界存在的普遍規律,因此,這一規律也可以應用到股價的反彈或下跌的預測中。某股票已經連續下跌一段時間,開始反彈。那麼反彈到哪裡會戛然而止呢?通常下跌深度和反彈的高度符合黃金分割比例:0.382,0.5,0.618,0.809等。 手工計算太麻煩,寫了個小軟體來計算相應的反彈位置。如圖: ...
  • php面試題之一——PHP核心技術(高級部分) 一、PHP核心技術 1.寫出一個能創建多級目錄的PHP函數(新浪網技術部) <?php /** * 創建多級目錄 * @param $path string 要創建的目錄 * @param $mode int 創建目錄的模式,在windows下可忽略 ...
  • 方法一 方法二 1 echo (strtotime('2007316')-strtotime('2007306'))/(24*3600); // strtotime,將一個英式日期戳轉換成一個unix時間戳,只要格式符合日期格式即可,註意後面6號的時候不能直接寫6.需要寫06,任何情況下不能寫6,否 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...