array_reduce 的理解
看 laravel 时遇到一个函数,琢磨了半天也没有理解,最后还是查了下资料才完全理解。
这里还是再做下笔记加深下印象。
php官方是这么写的:
array_reduce(array $array, callable $callback, [mixed $initial = null ])
什么意思呢,先举个简单点儿的栗子:
<?php
function sum($carry, $item) {
var_dump($carry, $item);
$carry += $item;
echo '<br><hr>';
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 '化妆打扮', '<br>';;
$next();
}
}
$firstSlice = function (){
echo '我要出去玩了~', '<br>';
};
$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 '穿上裙子', '<br>';
$next();
}
}
$arr = [
'MakeUp',
'Skirt'
];
那么执行过程会变成什么样呢?为了看清楚执行过程,先在 getSlice 中加上打印:
function getSlice(){
return function ($stack, $pipe){
echo '<pre>';
echo 'stack :';
var_dump($stack);
echo '<hr>';
echo 'pipe :';
var_dump($pipe);
echo '</pre>';
echo '<br><hr>';
return function () use ($stack, $pipe){
return $pipe::handle($stack);
};
};
}
执行结果
可以看到第一次执行时,stack 是一个匿名函数 , 也对应了官网介绍的作为第一次迭代时,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,你大神就是你大神,不能不服气。