<table id="caao6"><object id="caao6"></object></table>
  • <small id="caao6"><label id="caao6"></label></small>
    <acronym id="caao6"><rt id="caao6"></rt></acronym>
  • <acronym id="caao6"></acronym>
  • 淺談PHP匿名函數和閉包

     更新時間:2019-03-08 18:00:26   作者:佚名   我要評論(0)

    概述
    閉包和匿名函數在PHP 5.3.0中引入,這兩個特性非常有用,每個PHP開發者都應該掌握。
    匿名函數其實就是沒有名稱的函數,匿名函數可以賦值給變量,還能像其

    概述

    閉包和匿名函數在PHP 5.3.0中引入,這兩個特性非常有用,每個PHP開發者都應該掌握。

    匿名函數其實就是沒有名稱的函數,匿名函數可以賦值給變量,還能像其他任何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方法:

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

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

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

    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匿名函數和閉包詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網站的支持!

    您可能感興趣的文章:

    • 淺析PHP中的閉包和匿名函數
    • php的閉包(Closure)匿名函數初探
    • php的閉包(Closure)匿名函數詳解
    • PHP中的閉包(匿名函數)淺析

    相關文章

    • 淺談PHP匿名函數和閉包

      淺談PHP匿名函數和閉包

      概述 閉包和匿名函數在PHP 5.3.0中引入,這兩個特性非常有用,每個PHP開發者都應該掌握。 匿名函數其實就是沒有名稱的函數,匿名函數可以賦值給變量,還能像其
      2019-03-08
    • Node.js Stream ondata觸發時機與順序的探索

      Node.js Stream ondata觸發時機與順序的探索

      上次寫Stream pipe細節時,在源碼中發現一段無用邏輯,由此引發了對Stream data事件觸發時機與順序的探索。 無用邏輯 當時研究pipe細節是基于Node.js v8.11.1
      2019-03-08
    • PHP多進程通信-消息隊列使用

      PHP多進程通信-消息隊列使用

      向消息隊列發送數據和獲取數據的測試 <&#63;php $key=ftok(__FILE__,'a'); //獲取消息隊列 $queue=msg_get_queue($key,0666); //發送消息 //msg_send($que
      2019-03-08
    • spring security CSRF防護的示例代碼

      spring security CSRF防護的示例代碼

      CSRF是指跨站請求偽造(Cross-site request forgery),是web常見的攻擊之一。 從Spring Security 4.0開始,默認情況下會啟用CSRF保護,以防止CSRF攻擊應
      2019-03-08
    • Node.js Event Loop各階段講解

      Node.js Event Loop各階段講解

      Event Loop階段描述圖 timers timer階段處理setTimeout于setInterval回調,開始處理的時機與poll階段有關聯。 pending callbacks 該階段執行某些系統操作的回
      2019-03-08
    • 淺談PHP進程管理

      淺談PHP進程管理

      這篇文章是對之前一篇文章的補充和改進, 創建一個主(master)進程,主進程安裝定時器,每隔5分鐘檢測一次隊列長度,根據隊列長度計算需要的worker進程, 然后創
      2019-03-08
    • 淺談python的深淺拷貝以及fromkeys的用法

      淺談python的深淺拷貝以及fromkeys的用法

      1.join()的用法:使用前面的字符串.對后面的列表進行拼接,拼接結果是一個字符串 # lst = ["alex","dsb",'wusir','xsb'] # s = "".join(lst) # print(s) #ale
      2019-03-08
    • 淺談PHPANALYSIS提取關鍵字

      淺談PHPANALYSIS提取關鍵字

      最近在開發一個文章模塊功能,設計那邊提出要給文章生成對應標簽,用于文章關聯推送,這里和大家分享一下實現過程; 這里需要用到PHPAnalysis,下載鏈接如下
      2019-03-08
    • PHP htmlspecialchars()函數用法與實例講解

      PHP htmlspecialchars()函數用法與實例講解

      PHP htmlspecialchars() 函數 實例 把預定義的字符 "<" (小于)和 ">" (大于)轉換為 HTML 實體: <&#63;php $str = "This is some <b>bold</b> text.";
      2019-03-08
    • PHP lcfirst()函數定義與用法

      PHP lcfirst()函數定義與用法

      PHP lcfirst() 函數 實例 把 "Hello" 的首字符轉換為小寫。: <&#63;php echo lcfirst("Hello world!"); &#63;> 定義和用法 lcfirst()函數把字符串中的首
      2019-03-08

    最新評論

    种子磁力搜索器