萬盛學電腦網

 萬盛學電腦網 >> 網絡編程 >> php編程 >> php 5.3 閉包語法介紹 function() use() {}

php 5.3 閉包語法介紹 function() use() {}

PHP 5.3 加入了閉包語法,也就是匿名函數,允許開發者申明行內函數和在變量中保存。雖然這個語法和JavaScript的閉包相比有點怪異,但是對於PHP語言來說,這是一個良好的補充  代碼如下 復制代碼 <?php
/**
* 下面提到的代碼在PHP5.3以上版本運行通過.
*/
function callback($callback) {
$callback();
}
//輸出: This is a anonymous function.<br />/n
//這裡是直接定義一個匿名函數進行傳遞, 在以往的版本中, 這是不可用的.
//現在, 這種語法非常舒服, 和javascript語法基本一致, 之所以說基本呢, 需要繼續向下看
//結論: 一個舒服的語法必然會受歡迎的.
callback(function() {
print "This is a anonymous function.<br />/n";
});
//輸出: This is a closure use string value, msg is: Hello, everyone.<br />/n
//這裡首先定義了一個閉包, 這次戶口本上有名字了...
//use, 一個新鮮的家伙...
//眾所周知, 閉包: 內部函數使用了外部函數中定義的變量.
//在PHP新開放的閉包語法中, 我們就是用use來使用閉包外部定義的變量的.
//這裡我們使用了外部變量$msg, 定義完之後, 又對其值進行了改變, 閉包被執行後輸出的是原始值
//結論: 以傳值方式傳遞的基礎類型參數, 閉包use的值在閉包創建是就確定了.
$msg = "Hello, everyone";
$callback = function () use ($msg) {
print "This is a closure use string value, msg is: $msg. <br />/n";
};
$msg = "Hello, everybody";
callback($callback);
//輸出: This is a closure use string value lazy bind, msg is: Hello, everybody.<br />/n
//換一種引用方式, 我們使用引用的方式來use
//可以發現這次輸出是閉包定義後的值...
//這個其實不難理解, 我們以引用方式use, 那閉包use的是$msg這個變量的地址
//當後面對$msg這個地址上的值進行了改變之後, 閉包內再輸出這個地址的值時, 自然改變了.
$msg = "Hello, everyone";
$callback = function () use (&$msg) {
print "This is a closure use string value lazy bind, msg is: $msg. <br />/n";
};
$msg = "Hello, everybody";
callback($callback);
//輸出: This is a closure use object, msg is: Hello, everyone.<br />/n
//閉包中輸出的是之前被拷貝的值為Hello, everyone的對象, 後面是對$obj這個名字的一個重新賦值.
//可以這樣考慮
//1. obj是對象Hello, everyone的名字
//2. 對象Hello, everyone被閉包use, 閉包產生了一個對Hello, everyone對象的引用
//3. obj被修改為Hello, everybody這個對象的名字
//4. 注意, 是名字obj代表的實體變了, 而不是Hello, everyone對象, 那自然閉包的輸出還是前面的Hello, everyone
$obj = (object) "Hello, everyone";
$callback = function () use ($obj) {
print "This is a closure use object, msg is: {$obj->scalar}. <br />/n";
};
$obj = (object) "Hello, everybody";
callback($callback);
//輸出: This is a closure use object, msg is: Hello, everybody.<br />/n
//還是按照上面的步驟, 按部就班的來吧:
//1. obj名字指向Hello, everyone對象
//2. 閉包產生一個引用指向Hello, everyone對象
//3. 修改obj名字指向的對象(即Hello, everyone對象)的scalar值
//4. 執行閉包, 輸出的自然是Hello, everybody, 因為其實只有一個真正的對象
$obj = (object) "Hello, everyone";
$callback = function () use ($obj) {
print "This is a closure use object, msg is: {$obj->scalar}. <br />/n";
};
$obj->scalar = "Hello, everybody";
callback($callback);
//輸出: This is a closure use object lazy bind, msg is: Hello, everybody.<br />/n
//閉包引用的是什麼呢? &$obj, 閉包產生的引用指向$obj這個名字所指向的地址.
//因此, 無論obj怎麼變化, 都是逃不脫的....
//所以, 輸出的就是改變後的值
$obj = (object) "Hello, everyone";
$callback = function () use (&$obj) {
print "This is a closure use object lazy bind, msg is: {$obj->scalar}. <br />/n";
};
$obj = (object) "Hello, everybody";
callback($callback);
/**
* 一個利用閉包的計數器產生器
* 這裡其實借鑒的是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";
?>
copyright © 萬盛學電腦網 all rights reserved