溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在PHP中實現匿名函數和閉包

發布時間:2021-06-07 16:29:40 來源:億速云 閱讀:201 作者:Leah 欄目:開發技術

怎么在PHP中實現匿名函數和閉包?相信很多沒有經驗的人對此束手無策,為此本文總結了問題出現的原因和解決方法,通過這篇文章希望你能解決這個問題。

創建匿名函數

創建匿名函數很簡單:

//將匿名函數賦給一個變量,通過變量名+()的形式來調用
$greet = function () {
 return "Hello World";
};

echo $greet();

結果打印:

Hello World

匿名函數和普通的PHP函數很像:常用的句法相同,也接受參數,而且能返回值。不過閉包沒有函數名。

注:我們之所以能調用$greet變量,是因為這個變量的值是一個閉包,而且閉包對象實現了__invoke()魔術方法,只要變量名后有(),PHP就會查找并調用__invoke方法。

我們通常把匿名函數當做函數或方法的回調使用,事實上,很多PHP函數都會用到匿名函數,比如array_mappreg_replace_callback,這是使用PHP匿名函數的絕佳時機。記住,閉包和其他值一樣,可以作為參數傳入其他PHP函數:

$numberPlusOne = array_map(function ($number) {
 return $number += 1;
}, [1, 2, 3]);

print_r($numberPlusOne);

在匿名函數出現之前,要實現這樣的功能,PHP開發者只能單獨創建具名函數,然后使用名稱引用這個函數:

function incrementNumber ($number) {
 return $number += 1;
}

$numberPlusOne = array_map(‘incrementNumber', [1, 2, 3]);
print_r($numberPlusOne);

這樣做把回調的實現和使用場所隔離開了,而且使用閉包實現代碼更加簡潔。

創建閉包

包含自由變量的函數與為所有這些自由變量提供了變量綁定的環境一起,被稱為閉包。

function makeHelloWorld($name) { 
 $i = 0;
 return function()use($name, &$i){
  echo $name.$i. ' <br>';
  $i++;
 };

}
$hello1 = makeHelloWorld("itbsl");
$hello2 = makeHelloWorld("kevin");
$hello1();
$hello1();
$hello1();
$hello2();

打印結果:

itbsl0
itbsl1
itbsl2
kevin0

從父作用域繼承變量

在PHP中必須手動調用閉包對象的bindTo方法或使用use關鍵字把父作用域的變量及狀態附加到PHP閉包中。而實際應用中,又以使用use關鍵字實現居多。

use關鍵字

實際上,Laravel框架中也大量使用了閉包,最常見的比如路由定義:

Route::group(['domain' => '{account}.myapp.com'], function () {
 Route::get('user/{id}', function ($account, $id) {
  //
 });
});

這里面的兩個function都是匿名函數。而從父作用域繼承變量的使用場景在Laravel底層源碼中也是俯拾即是,比如Model.php(Illuminate\Database\Eloquent)的saveOrFail方法:

怎么在PHP中實現匿名函數和閉包

該方法的作用是使用事務將模型數據保存到數據庫,這里面我們使用匿名函數返回保存狀態,同時使用use關鍵字將父作用域的$options傳遞給該閉包以便其能夠訪問這個數據。

此外,還支持傳遞多個父作用域變量到匿名函數,比如還是在Model類中的forceFill方法:

怎么在PHP中實現匿名函數和閉包

多個變量以逗號分隔即可。

bindTo方法

我們在前面已經提到,閉包是一個對象,所以我們可以在閉包中使用$this關鍵字獲取閉包的內部狀態,閉包對象的默認狀態沒什么用,需要注意的是其中的__invoke魔術方法和bindTo方法。

__invoke的作用前面已經說過,當嘗試以調用函數的方式調用一個對象時,__invoke() 方法會被自動調用。

接下來我們來看看bindTo方法,通過該方法,我們可以把閉包的內部狀態綁定到其他對象上。這里bindTo方法的第二個參數顯得尤為重要,其作用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就可以在其他地方訪問邦定閉包的對象中受保護和私有的成員變量。

你會發現,PHP框架經常使用bindTo方法把路由URL映射到匿名回調函數上,框架會把匿名回調函數綁定到應用對象上,這樣在匿名函數中就可以使用$this關鍵字引用重要的應用對象:

class App {
 protected $routes = [];
 protected $responseStatus = '200 OK';
 protected $responseContentType = 'text/html';
 protected $responseBody = 'Laravel學院';

 public function addRoute($routePath, $routeCallback) {
  $this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);
 }

 public function dispatch($currentPath) {
  foreach ($this->routes as $routePath => $callback) {
   if( $routePath === $currentPath) {
    $callback();
   }
  }
  header('HTTP/1.1 ' . $this->responseStatus);
  header('Content-Type: ' . $this->responseContentType);
  header('Content-Length: ' . mb_strlen($this->responseBody));
  echo $this->responseBody;
 }

}

這里我們需要重點關注addRoute方法,這個方法的參數分別是一個路由路徑和一個路由回調,dispatch方法的參數是當前HTTP請求的路徑,它會調用匹配的路由回調。第9行是重點所在,我們將路由回調綁定到了當前的App實例上。這么做能夠在回調函數中處理App實例的狀態:

$app = new App();
$app->addRoute(‘user/nonfu', function(){
 $this->responseContentType = ‘application/json;charset=utf8';
 $this->responseBody = ‘{“name”:”LaravelAcademy"}';
});
$app->dispatch(‘user/nonfu');
在Larval底層也有用到bindTo方法,詳見Illuminate\Support\Traits\Macroable的__call方法:

看完上述內容,你們掌握怎么在PHP中實現匿名函數和閉包的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

php
AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女