看 laravel 时遇到一个函数,琢磨了半天也没有理解,最后还是查了下资料才完全理解。
这里还是再做下笔记加深下印象。
php官方是这么写的:
array_reduce(array $array, callable $callback, [mixed $initial = null ])
什么意思呢,先举个简单点儿的栗子:
function sum($carry, $item) {
var_dump($carry, $item);
$carry += $item;
echo '
';
return $carry;
}
$a = array(1, 2, 3, 4, 5);
var_dump(array_reduce($a, 'sum', 10));
输出结果为:
int(10) int(1)
int(11) int(2)
int(13) int(3)
int(16) int(4)
int(20) int(5)
int(25)
可以看出 array_reduce 的第三个参数传给 callable sum 并作为第一个参数,然后 array $a 的第一个元素作为第二个参数,即sum(10, 1),然后把计算结果返回给下一次迭代即 sum(11, 2),依次类推直至程序结束。
简单的理解了,再来点复杂的,不过在开始之前,先来熟悉下匿名函数:
匿名函数:
$name = 'well';
$greet = function () use ($name){
echo 'hello ' ,$name;
};
$greet ();
# hello well
function closureFunc($name){
$func = function() use ($name){
echo "hello" ,$name;
};
$func();
}
closureFunc('well');
# hello well
接下来就是重点了,在 array_reduce 中使用匿名函数,举个例子,西瓜姑娘接到闺蜜电话出去玩,肯定要先化妆打扮一下再出门:
class MakeUp {
public static function handle(Closure $next) {
echo '化妆打扮', '
';;
$next();
}
}
$firstSlice = function (){
echo '我要出去玩了~', '
';
};
$arr = [
'MakeUp'
];
function getSlice(){
return function ($stack, $pipe){
return function () use ($stack, $pipe){
return $pipe::handle($stack);
};
};
}
$go = array_reduce($arr, getSlice(), $firstSlice);
$go();
输出结果为:
化妆打扮
我要出去玩了~
这段程序执行时首先运行 getSlice($firstSlice, 'DressUp'),然后拼装执行 MakeUp::handle(),
MakeUp::handle() 中先去化妆打扮,然后再通过匿名Closure $next执行 $firstSlice 出门去玩。
如果觉得仅仅是化妆还不够,还要再穿条美美的裙子出门,那么再对上边的处理进行改造,添加穿裙子的过程:
class Skirt {
public static function handle(Closure $next) {
echo '穿上裙子', '
';
$next();
}
}
$arr = [
'MakeUp',
'Skirt'
];
那么执行过程会变成什么样呢?为了看清楚执行过程,先在 getSlice 中加上打印:
function getSlice(){
return function ($stack, $pipe){
echo '
';
echo 'stack :';
var_dump($stack);
echo '
';
echo 'pipe :';
var_dump($pipe);
echo '
';echo '
';
return function () use ($stack, $pipe){
return $pipe::handle($stack);
};
};
}
执行结果
可以看到第一次执行时,stack 是一个匿名函数 , 也对应了官网介绍的作为第一次迭代时,$carry 是 initial,也就是$firstSlice。pipe 为MakeUp,但是它并没有执行而是直接返回了,也就是直接返回
function ($stack, $pipe){
return function () use ($stack, $pipe){
return $pipe::handle($stack);
};
匿名函数 function ($stack, $pipe)被返回后,也就相当于一个 array_reduce($arr, getSlice(), $firstSlice);,不同的是匿名函数 function ($stack, $pipe)代替了$firstSlice被传给下一次迭代。
第二次迭代时执行 Skire::handle,先穿上了美美的裙子,然后$next 执行匿名函数 function ($stack, $pipe),也就是去MakeUp::handle,化妆后继续执行 $next,也就是匿名函数 $firstSlice。
emmm,你大神就是你大神,不能不服气。