匿名函數(shù)
提到閉包就不得不想起匿名函數(shù),也叫閉包函數(shù)(closures),貌似PHP閉包實現(xiàn)主要就是靠它。聲明一個匿名函數(shù)是這樣:
代碼如下:
$func = function() {
}; //帶結(jié)束符
可以看到,匿名函數(shù)因為沒有名字,如果要使用它,需要將其返回給一個變量。匿名函數(shù)也像普通函數(shù)一樣可以聲明參數(shù),調(diào)用方法也相同:
代碼如下:
$func = function( $param ) {
echo $param;
};
$func( 'some string' );
//輸出:
//some string
順便提一下,PHP在引入閉包之前,也有一個可以創(chuàng)建匿名函數(shù)的函數(shù):create function,但是代碼邏輯只能寫成字符串,這樣看起來很晦澀并且不好維護,所以很少有人用。
實現(xiàn)閉包
將匿名函數(shù)在普通函數(shù)中當(dāng)做參數(shù)傳入,也可以被返回。這就實現(xiàn)了一個簡單的閉包。
下面請看例子
//例一
//在函數(shù)里定義一個匿名函數(shù),并且調(diào)用它
function printStr() {
??? $func = function( $str ) {
??????? echo $str;
??? };
??? $func( 'some string' );
}
printStr();
輸出: some string
//例三
//把匿名函數(shù)當(dāng)做參數(shù)傳遞,并且調(diào)用它
function callFunc( $func ) {
? ? $func( 'some string' );
}
$printStrFunc = function( $str ) {
? ? echo $str;
};
// print_r(callFunc( $printStrFunc ));
// //也可以直接將匿名函數(shù)進行傳遞。如果你了解js,這種寫法可能會很熟悉
callFunc( function( $str ) {
? ? echo $str;
} );
輸出some string
連接閉包和外界變量的關(guān)鍵字:USE
閉包可以保存所在代碼塊上下文的一些變量和值。PHP在默認情況下,匿名函數(shù)不能調(diào)用所在代碼塊的上下文變量,而需要通過使用use關(guān)鍵字。
function getMoney() {
? ? $rmb = 1;
? ? $dollar = 6;
? ? $func = function() use ( $rmb ) {
? ? ? ? echo $rmb;
? ? ? ? echo $dollar;
? ? };
? ? $func();
}
getMoney();
//輸出:
//1
//報錯,找不到dorllar變量
可以看到,dollar沒有在use關(guān)鍵字中聲明,在這個匿名函數(shù)里也就不能獲取到它,所以開發(fā)中要注意這個問題。
有人可能會想到,是否可以在匿名函數(shù)中改變上下文的變量,但我發(fā)現(xiàn)是不可以的:
function getMoney() {
??? $rmb = 1;
??? $func = function() use ( $rmb ) {
??????? echo $rmb;
??????? //把$rmb的值加1
??????? $rmb++;
??? };
??? $func();
??? echo $rmb;
}
getMoney();
//輸出:
//1
//1
啊,原來use所引用的也只不過是變量的一個副本而已。但是我想要完全引用變量,而不是復(fù)制。
要達到這種效果,其實在變量前加一個 & 符號就可以了:
function getMoneyFunc() {
??? $rmb = 1;
??? $func = function() use ( &$rmb ) {
??????? echo $rmb;
??????? //把$rmb的值加1
??????? $rmb++;
??? };
??? return $func;
}
$getMoney = getMoneyFunc();
$getMoney();
$getMoney();
$getMoney();
//輸出:
//1
//2
//3
PHP閉包的特性并沒有太大驚喜,其實用CLASS就可以實現(xiàn)類似甚至強大得多的功能,更不能和js的閉包相提并論,只能期待PHP以后對閉包支持的改進。不過匿名函數(shù)還是挺有用的,比如在使用preg_replace_callback等之類的函數(shù)可以不用在外部聲明回調(diào)函數(shù)了。