匿名函數(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();
//例二
//在函數(shù)中把匿名函數(shù)返回性宏,并且調(diào)用它
function getPrintStrFunc() {
$func = function( $str ) {
echo $str;
};
return $func;
}
$printStrFunc = getPrintStrFunc();
$printStrFunc( 'some string' );
//例三
//把匿名函數(shù)當(dāng)做參數(shù)傳遞群井,并且調(diào)用它
function callFunc( $func ) {
$func( 'some string' );
}
$printStrFunc = function( $str ) {
echo $str;
};
callFunc( $printStrFunc );
//也可以直接將匿名函數(shù)進行傳遞。如果你了解js衔沼,這種寫法可能會很熟悉
callFunc(
function( $str ) {
echo $str;
}
);
連接閉包和外界變量的關(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 getMoney() {
$rmb = 1;
$func = function() use ( &$rmb ) {
echo $rmb;
//把$rmb的值加1
$rmb++;
};
$func();
echo $rmb;
}
getMoney();
//輸出:
//1
//2
好傍衡,這樣匿名函數(shù)就可以引用上下文的變量了。如果將匿名函數(shù)返回給外界负蠕,匿名函數(shù)會保存use所引用的變量蛙埂,而外界則不能得到這些變量,這樣形成‘閉包’這個概念可能會更清晰一些遮糖。
根據(jù)描述改變一下上面的例子:
function getMoneyFunc() {
$rmb = 1;
$func = function() use ( &$rmb ) {
echo $rmb;
//把$rmb的值加1
$rmb++;
};
return $func;
}
$getMoney = \getMoneyFunc();
$getMoney();
$getMoney();
$getMoney();
//輸出:
//1
//2
//3
總結(jié)
PHP閉包的特性并沒有太大驚喜绣的,其實用CLASS就可以實現(xiàn)類似甚至強大得多的功能,更不能和js的閉包相提并論欲账,只能期待PHP以后對閉包支持的改進屡江。不過匿名函數(shù)還是挺有用的,比如在使用preg_replace_callback等之類的函數(shù)可以不用在外部聲明回調(diào)函數(shù)了赛不。