参考地址 array_reduce 的理解
看 laravel 时遇到一个函数,琢磨了半天也没有理解,最后还是查了下资料才完全理解。
这里还是再做下笔记加深下印象。
php官方是这么写的:
array_reduce(array $array, callable $callback, [mixed $initial = null ])
什么意思呢,先举个简单点儿的栗子:
<?phpfunction 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 是一个匿名函数 , 也对应了官网介绍的作为第一次迭代时,$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,你大神就是你大神,不能不服气。