php閉包(Closure),匿名函數執行個體詳解

來源:互聯網
上載者:User
本文主要給大家介紹的是php5.3引入的PHP匿名函數,也就是閉包(Closure),以及閉包的作用,非常詳細,這裡推薦給有需要的小夥伴們。

php的閉包(Closure)也就是匿名函數,是PHP5.3引入的。

閉包的文法很簡單,需要注意的關鍵字就只有use,use是串連閉包和外界變數。

代碼如下:

$a = function() use($b) {}

簡單例子如下:

代碼如下:

function callback($fun) {$fun();}$msg = "Hello, everyone";$fun = function () use($msg) {print "This is a closure use string value, msg is: $msg. <br />/n";};$msg = "Hello, everybody";callback($fun);

結果是:This is a closure use string value, msg is: Hello, everyone. <br />/n

在PHP新開放的閉包文法中, 我們用use來使用閉包外部定義的變數的。這裡我們使用了外部變數$msg,定義完之後,又對其值進行了改變,閉包被執行後輸出的是原始值。以傳值方式傳遞的基礎型別參數,閉包use的值在閉包建立是就確定了。

小應用如下:

代碼如下:

/**  * 一個利用閉包的計數器產生器  * 這裡其實借鑒的是python中介紹閉包時的例子...  * 我們可以這樣考慮:  *      1. counter函數每次調用, 建立一個局部變數$counter, 初始化為1.  *      2. 然後建立一個閉包, 閉包產生了對局部變數$counter的引用.  *      3. 函數counter返回建立的閉包, 並銷毀局部變數, 但此時有閉包對$counter的引用,   *          它並不會被回收, 因此, 我們可以這樣理解, 被函數counter返回的閉包, 攜帶了一個游離態的  *          變數.  *      4. 由於每次調用counter都會建立獨立的$counter和閉包, 因此返回的閉包相互之間是獨立的.  *      5. 執行被返回的閉包, 對其攜帶的游離態變數自增並返回, 得到的就是一個計數器.  * 結論: 此函數可以用來產生相互獨立的計數器.  */  function counter() {      $counter = 1;      return function() use(&$counter) {return $counter ++;};  }  $counter1 = counter();  $counter2 = counter();  echo "counter1: " . $counter1() . "<br />/n";  echo "counter1: " . $counter1() . "<br />/n";  echo "counter1: " . $counter1() . "<br />/n";  echo "counter1: " . $counter1() . "<br />/n";  echo "counter2: " . $counter2() . "<br />/n";  echo "counter2: " . $counter2() . "<br />/n";  echo "counter2: " . $counter2() . "<br />/n";  echo "counter2: " . $counter2() . "<br />/n";  ?>

閉包的作用

1. 減少foreach的迴圈的代碼
比如手冊http://php.net/manual/en/functions.anonymous.php 中的例子Cart

代碼如下:

<?php// 一個基本的購物車,包括一些已經添加的商品和每種商品的數量。// 其中有一個方法用來計算購物車中所有商品的總價格。該方法使用了一個closure作為回呼函數。class Cart{    const PRICE_BUTTER  = 1.00;    const PRICE_MILK    = 3.00;    const PRICE_EGGS    = 6.95;    protected   $products = array();    public function add($product, $quantity)    {        $this->products[$product] = $quantity;    }    public function getQuantity($product)    {        return isset($this->products[$product]) ? $this->products[$product] :               FALSE;    }    public function getTotal($tax)    {        $total = 0.00;        $callback =            function ($quantity, $product) use ($tax, &$total)            {                $pricePerItem = constant(CLASS . "::PRICE_" .                    strtoupper($product));                $total += ($pricePerItem * $quantity) * ($tax + 1.0);            };        //使用使用者自訂函數對數組中的每個元素做回調處理        array_walk($this->products, $callback);        return round($total, 2);;    }}$my_cart = new Cart;// 往購物車裡添加條目$my_cart->add('butter', 1);$my_cart->add('milk', 3);$my_cart->add('eggs', 6);// 打出出總價格,其中有 5% 的銷售稅.print $my_cart->getTotal(0.05) . "\n";// The result is 54.29?>

這裡如果我們改造getTotal函數必然要使用到foreach。

2. 減少函數的參數

代碼如下:

function html($code , $id="", $class=""){if ($id !== "") $id = " id = \"$id\"" ;$class = ($class !== "")? " class =\"$class\">":">";$open = "<$code$id$class";$close = "</$code>";return function ($inner = "") use ($open, $close){return "$open$inner$close";    };}

如果是使用平時的方法,我們會把inner放到html函數參數中,這樣不管是代碼閱讀還是使用都不如使用閉包。

3. 解除遞迴函式

代碼如下:

<?php$fib = function($n) use(&$fib) {    if($n == 0 || $n == 1) return 1;    return $fib($n - 1) + $fib($n - 2);};echo $fib(2) . "\n"; // 2$lie = $fib;$fib = function(){die('error');};//rewrite $fib variable echo $lie(5); // error   because $fib is referenced by closure

注意上題中的use使用了&,這裡不使用&會出現錯誤fib(n-1)是找不到function的(前面沒有定義fib的類型)

所以想使用閉包解除迴圈函數的時候就需要使用

代碼如下:

<?php$recursive = function () use (&$recursive){// The function is now available as $recursive}

這樣的形式。

4. 延遲綁定

如果你需要延遲綁定use裡面的變數,你就需要使用引用,否則在定義的時候就會做一份拷貝放到use中

代碼如下:

<?php$result = 0;$one = function(){    var_dump($result);};$two = function() use ($result){    var_dump($result);};$three = function() use (&$result){    var_dump($result);};$result++;$one();    // outputs NULL: $result is not in scope$two();    // outputs int(0): $result was copied$three();    // outputs int(1)

使用引用和不使用引用就代表了是調用時賦值,還是申明時候賦值

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.