一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - PHP教程 - 淺談PHP匿名函數(shù)和閉包

淺談PHP匿名函數(shù)和閉包

2019-06-03 11:07itbsl服務(wù)器之家 PHP教程

這篇文章主要介紹了PHP匿名函數(shù)和閉包,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

概述

閉包和匿名函數(shù)在PHP 5.3.0中引入,這兩個(gè)特性非常有用,每個(gè)PHP開發(fā)者都應(yīng)該掌握。

匿名函數(shù)其實(shí)就是沒有名稱的函數(shù),匿名函數(shù)可以賦值給變量,還能像其他任何PHP函數(shù)對象那樣傳遞。不過匿名函數(shù)仍然是函數(shù),因此可以調(diào)用,還可以傳入?yún)?shù),適合作為函數(shù)或方法的回調(diào)。

閉包是指在創(chuàng)建時(shí)封裝周圍狀態(tài)的函數(shù),即使閉包所在的環(huán)境的不存在了,閉包中封裝的狀態(tài)依然存在。

創(chuàng)建匿名函數(shù)

創(chuàng)建匿名函數(shù)很簡單:

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

echo $greet();

結(jié)果打印:

Hello World

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

注:我們之所以能調(diào)用$greet變量,是因?yàn)檫@個(gè)變量的值是一個(gè)閉包,而且閉包對象實(shí)現(xiàn)了__invoke()魔術(shù)方法,只要變量名后有(),PHP就會(huì)查找并調(diào)用__invoke方法。

我們通常把匿名函數(shù)當(dāng)做函數(shù)或方法的回調(diào)使用,事實(shí)上,很多PHP函數(shù)都會(huì)用到匿名函數(shù),比如array_mappreg_replace_callback,這是使用PHP匿名函數(shù)的絕佳時(shí)機(jī)。記住,閉包和其他值一樣,可以作為參數(shù)傳入其他PHP函數(shù):

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

print_r($numberPlusOne);

在匿名函數(shù)出現(xiàn)之前,要實(shí)現(xiàn)這樣的功能,PHP開發(fā)者只能單獨(dú)創(chuàng)建具名函數(shù),然后使用名稱引用這個(gè)函數(shù):

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

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

這樣做把回調(diào)的實(shí)現(xiàn)和使用場所隔離開了,而且使用閉包實(shí)現(xiàn)代碼更加簡潔。

創(chuàng)建閉包

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

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();

打印結(jié)果:

itbsl0 
itbsl1 
itbsl2 
kevin0

從父作用域繼承變量

在PHP中必須手動(dòng)調(diào)用閉包對象的bindTo方法或使用use關(guān)鍵字把父作用域的變量及狀態(tài)附加到PHP閉包中。而實(shí)際應(yīng)用中,又以使用use關(guān)鍵字實(shí)現(xiàn)居多。

use關(guān)鍵字

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

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

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

 

淺談PHP匿名函數(shù)和閉包
 

 

該方法的作用是使用事務(wù)將模型數(shù)據(jù)保存到數(shù)據(jù)庫,這里面我們使用匿名函數(shù)返回保存狀態(tài),同時(shí)使用use關(guān)鍵字將父作用域的$options傳遞給該閉包以便其能夠訪問這個(gè)數(shù)據(jù)。

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

 

淺談PHP匿名函數(shù)和閉包

 

多個(gè)變量以逗號(hào)分隔即可。

bindTo方法

我們在前面已經(jīng)提到,閉包是一個(gè)對象,所以我們可以在閉包中使用$this關(guān)鍵字獲取閉包的內(nèi)部狀態(tài),閉包對象的默認(rèn)狀態(tài)沒什么用,需要注意的是其中的__invoke魔術(shù)方法和bindTo方法。

__invoke的作用前面已經(jīng)說過,當(dāng)嘗試以調(diào)用函數(shù)的方式調(diào)用一個(gè)對象時(shí),__invoke() 方法會(huì)被自動(dòng)調(diào)用。

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

你會(huì)發(fā)現(xiàn),PHP框架經(jīng)常使用bindTo方法把路由URL映射到匿名回調(diào)函數(shù)上,框架會(huì)把匿名回調(diào)函數(shù)綁定到應(yīng)用對象上,這樣在匿名函數(shù)中就可以使用$this關(guān)鍵字引用重要的應(yīng)用對象:

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

 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;
 }

}

這里我們需要重點(diǎn)關(guān)注addRoute方法,這個(gè)方法的參數(shù)分別是一個(gè)路由路徑和一個(gè)路由回調(diào),dispatch方法的參數(shù)是當(dāng)前HTTP請求的路徑,它會(huì)調(diào)用匹配的路由回調(diào)。第9行是重點(diǎn)所在,我們將路由回調(diào)綁定到了當(dāng)前的App實(shí)例上。這么做能夠在回調(diào)函數(shù)中處理App實(shí)例的狀態(tài):

$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匿名函數(shù)和閉包詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對服務(wù)器之家網(wǎng)站的支持!

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 91麻豆国产福利在线观看 | 小夫妻天天恶战 | 亚洲国产在线99视频 | 日本高清视频在线观看 | 二次元美女挤奶漫画 | 免看一级a一片成人123 | 99久久伊人精品波多野结衣 | 国产精品视频久 | 亚洲男人天堂 | 狠狠色成人综合网图片区 | 日本高清中文字幕一区二区三区 | www.久久艹| 大奶妈咪女教师 | 91久久国产青草亚洲 | 日韩久久综合 | 99热这里只有精品在线观看 | 闺蜜调教我做她的脚奴 | 国产精品一区二区三区免费 | 亚洲精品无码不卡 | 王淑兰李思雨李铁柱乡村小说免费 | 精品推荐国产麻豆剧传媒 | 欧美3d怪物交videos网站 | 日本艳鉧动漫1~6在线观看 | 国产精品日韩欧美在线 | 精品综合久久久久久8888 | 日本免费一区二区三区a区 日本免费三片在线观看 | 久久亚洲精品AV成人无 | 久久精品一卡二卡三卡四卡视频版 | vod国产成人精品视频 | 久久水蜜桃亚洲AV无码精品偷窥 | 国产免费一区二区三区 | 手机能看的黄色网站 | 亚洲日本aⅴ片在线观看香蕉 | 国产精品亚洲专区在线播放 | 69japanese日本100 6969精品视频在线观看 | 久久成人永久免费播放 | jizzjizz3d动漫 | 亚洲邪恶天堂影院在线观看 | 亚洲色图亚洲色图 | 911福利视频 | 亚洲精品私拍国产福利在线 |