首頁 > 軟體

PHP中的閉包function() use() {}使用場景和技巧

2022-12-15 14:02:21

由於存在函數內部不能存取全域性作用的,所以就需要一種可以引入上一級作用域的語法結構,可以通過use使用函數宣告時所在作用域的變數的值。php的閉包可能不常用,但是在某些場合之下還是可以考慮用php的閉包來實現某些功能的。

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

提到閉包就不得不想起匿名函數,也叫閉包函數(closures)。匿名函數因為沒有名字,如果要使用它,需要將其返回給一個變數。匿名函數也像普通函數一樣可以宣告引數,呼叫方法也相同。

    $func = function( $param ) {
     echo $param;
    };
    $func( 'some string' );

    //輸出:
    //some string

function () use () {}語法結構

在PHP中,由於存在函數內部不能存取全域性作用的,所以就需要一種可以引入上一級作用域的語法結構,和上級作用域產生了聯絡,語句結構如下:

function () use () {}

將需要引入到這個函數作用於內的變數寫入到use的括號裡面就可以了,舉個例子

<?php
$a = 1;
$closure = function () use ($a) {
    echo $a;
};
$closure();
?>

輸出的結果為

1

閉包function() use() {}的使用場景:

1 減少foreach的迴圈的程式碼

<?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("::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使用了&,這裡不使用&會出現錯誤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)

使用參照和不使用參照就代表了是呼叫時賦值,還是申明時候賦值

到此這篇關於PHP中的閉包function() use() {}使用場景和技巧的文章就介紹到這了,更多相關PHP function() use()內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com