閉包函數(closures)
PHP閉包實現主要就是靠匿名函數
將匿名函數在普通函數中當做參數傳入,也可以被返回。這就實現了一個簡單的閉包。
通俗的說:子函數可以使用父函數中的局部變量,這種行為就叫做閉包!
閉包的兩個特點:
1、作為一個函數變量的一個引用 - 當函數返回時,其處于激活狀態。
2、一個閉包就是當一個函數返回時,一個沒有釋放資源的棧區。
其實上面兩點可以合成一點,就是閉包函數返回時,該函數內部變量處于激活狀態,函數所在棧區依然保留
例1
//在函數里定義一個匿名函數,并且調用它
function printStr() {
$func = function( $str ) {
echo $str;
};
$func( 'some string' );
}
printStr();
連接閉包和外界變量的關鍵字:USE
閉包可以保存所在代碼塊上下文的一些變量和值。PHP在默認情況下,匿名函數不能調用所在代碼塊的上下文變量,而需要通過使用use關鍵字。
例2
//在函數中把匿名函數返回,并且調用它
function getPrintStrFunc() {
$func = function( $str ) {
echo $str;
};
return $func;
}
$printStrFunc = getPrintStrFunc();
$printStrFunc( 'some string' );
例3
//把匿名函數當做參數傳遞,并且調用它
function callFunc( $func ) {
$func( 'some string' );
}
$printStrFunc = function( $str ) {
echo $str;
};
callFunc( $printStrFunc );
//也可以直接將匿名函數進行傳遞。這種寫法可能會很熟悉(js)
callFunc( function( $str ) {
echo $str;
} );
function getMoney() {
$rmb = 1;
$dollar = 6;
$func = function() use ( $rmb ) {
echo $rmb;
echo $dollar;
};
$func();
}
getMoney();
//輸出:
//1
//報錯,找不到dorllar變量
可以看到,dollar沒有在use關鍵字中聲明,在這個匿名函數里也就不能獲取到它,所以開發中要注意這個問題。
是否可以在匿名函數中改變上下文的變量---- 否
function getMoney() {
$rmb = 1;
$func = function() use ( $rmb ) {
echo $rmb;
//把$rmb的值加1
$rmb++;
};
$func();
echo $rmb;
}
getMoney();
//輸出:
//1
//1
原來use所引用的也只不過是變量的一個副本而已
但是我想要完全引用變量,而不是復制。要達到這種效果,其實在變量前加一個& 符號就可以了:
function getMoney() {
$rmb = 1;
$func = function() use ( &$rmb ) {
echo $rmb;
//把$rmb的值加1
$rmb++;
};
$func();
echo $rmb;
}
getMoney();
//輸出:
//1
//2
11
如果將匿名函數返回給外界,匿名函數會保存use所引用的變量,而外界則不能得到這些變量,這樣形成‘閉包'這個概念可能會更清晰一些。
function getMoneyFunc() {
$rmb = 1;
$func = function() use ( &$rmb ) {
echo $rmb;
//把$rmb的值加1
$rmb++;
};
return $func;
}
$getMoney = getMoneyFunc();
$getMoney();
$getMoney();
$getMoney();
//輸出:1 2 3
1,閉包外層是個函數.
2,閉包內部都有函數.
3,閉包會return內部函數.
4,閉包返回的函數內部不能有return.(因為這樣就真的結束了)
5,執行閉包后,閉包內部變量會存在,而閉包內部函數的內部變量不會存在.
閉包的應用場景
1、保護函數內的變量安全。以最開始的例子為例,外層函數中變量只有內部函數才能訪問,而無法通過其他途徑訪問到,因此保護了外層函數中變量的安全性。
2、在內存中維持一個變量。依然如前例,由于閉包,外層函數中的變量一直存在于內存中,因此每次執行,都會使用到。
建議:
PHP閉包的特性并沒有太大驚喜,其實用CLASS就可以實現類似甚至強大得多的功能,更不能和js的閉包相提并論,只能期待PHP以后對閉包支持的改進。不過匿名函數還是挺有用的,比如在使用 array_filter()等之類的函數可以不用在外部聲明回調函數了。
目前還不穩定,不適用于正式開發。
1.php
<?php
$a = 500;
function one() {
$a = 100;
echo "11111111111111111<br>";
function two() {
global $a;
echo "222222 {$a} 2222222<br>";
}
function three() {
echo "3333333333333333<br>";
}
two();
three();
}
one();
2.php
<?php
function demo() {
$a = 10;
$b = 20;
$one = function($str) use (&$a, &$b) {
echo $str."<br>";
echo $b."<br>";
$a++;
echo $a."<br>";
return "";
};
return $one;
}
$var=demo();
$var("hello world");
$var("this is a test");
$var("############");
test.php
<?php
function demo($fun) {
echo $fun();
}
demo(function(){
return "@2222222222222222<br>";
});
demo(function(){
return "###################<br>";
});