php 5.3新增的閉包語法介紹function() use() {}
阿新 • • 發佈:2018-01-21
vars .net details article body 初始 bar amp bin
[php] view plain copy
- <?php
- /**
- * author: selfimpr
- * mail: [email protected]
- * blog: http://blog.csdn.net/lgg201
- * 下面提到的代碼在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";
- ?>
php 5.3新增的閉包語法介紹function() use() {}