最近在學習workerman的時候比較頻繁的接觸到回調函數,使用中經常會因為worker的使用方式不同,會用這兩種不同的方式去調用外部的worker變數,這裡就整理一下PHP閉包獲取外部變數和global關鍵字聲明變數的區別。 閉包 閉包是一個常見的概念,我們通常可以將其與回調函數配合使用,可以使代 ...
最近在學習workerman的時候比較頻繁的接觸到回調函數,使用中經常會因為worker的使用方式不同,會用這兩種不同的方式去調用外部的worker變數,這裡就整理一下PHP閉包獲取外部變數和global關鍵字聲明變數的區別。
閉包
閉包是一個常見的概念,我們通常可以將其與回調函數配合使用,可以使代碼更加簡潔易讀。
閉包可以通過拷貝的方式讓函數使用父作用域中的變數。如:
$global = 'hello';
$bbb = function()use($global){
echo $global."\n";
};
$bbb();
//輸出 'hello'
global關鍵字聲明變數
通過global聲明變數同樣可以使函數體調用到函數外部的變數,不過global與use不同,globle關鍵字會使創建一個與外部變數同名的引用,並且在函數內對變數作出修改同樣會作用域外部變數。
$global = 'hello';
$fun = function(){
global $global;
$global =' world';
echo $global."\n";
};
$fun();
// 輸出 'world'
這裡只是創建一個同名引用而已,並不會改變原本外部變數$global的作用域,也就是說在另外一個函數中調用該依舊需要聲明或者使用閉包
$global = 'hello';
$fun = function(){
global $global;
$global =' world';
echo 'a:'.$global."\n";
};
$ccc = function(){
echo 'b:'.$global;
};
$fun()
$ccc()
/*
輸出
a: world
Notice: Undefined variable: global in xxxxxxx on line xx
*/
再稍微改一下代碼,這樣更容易對比閉包和global關鍵字聲明變數這兩種訪問外部變數方式的區別。
<?php
$global = 'hello';
$fun = function(){
global $global;
$global ='world';
echo 'a:'.$global."\n";
};
$bbb = function()use($global){
$global = 'china';
echo 'c:'.$global."\n";
};
$fun();
echo 'b:'.$global."\n";
$bbb();
echo 'd:'.$global;
這裡b和d兩個輸出可以看出來,global改變了外部變數的值,而閉包方式並沒有。
輸出:
a: world
b: world
c:china
d: world
最後再貼一個官方文檔中比較經典的使用匿名函數,閉包與回調函數配合的例子:
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);
print $my_cart->getTotal(0.05) . "\n";