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";
- ?>
新聞熱點
疑難解答