php array_reduce 的理解laravel

2020-02-19 23:06:46

参考地址 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 是一个匿名函数 , 也对应了官网介绍的作为第一次迭代时,$carryinitial,也就是$firstSlicepipeMakeUp,但是它并没有执行而是直接返回了,也就是直接返回

 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,你大神就是你大神,不能不服气。



  • 2020-03-14 16:35:00

    nuxt.js部署全过程(ubuntu+nginx+node+pm2)

    系统的话本篇是Ubuntu 16.04.6 ,centos也行,大同小异都是Linux。不过如果你是初学者,最好和我使用一样的,因为因为发行版本不同而导致的差异可能导致运行某些东西失败,找问题要找好久。windows server不推荐了,企业用的多,小服务器跑windows server比较费劲。

  • 2020-03-14 23:15:25

    icomoon使用详细介绍

    此篇博文讲述如何利用icomoon导入图标,从而把自己想要的都通过icomoon方式进行,大家都知道,网站以及移动端,用图标还是尽量选择这种。因为直接用image有些图标会失真,从而也是前端开发之中,需求去掌握的一项,很简单的就几个步骤。

  • 2020-03-14 23:39:59

    vuetify和@nuxt/vuetify icon 之我见

    vuetify中v-icon,貌似默认支持 Material Design Icons, Material Icons, Font Awesome 4 and Font Awesome 5, 我自己单独引入了vuetify 用哪一个图标都没有问题。但是用了@nuxt/vuetify只能用mdi-home这样的。不知道因为啥。肯定是封装后,封装成一个了。 但是我修改vuetify的设置,哪一个图标也都能用。哎,不过多研究了。

  • 2020-03-16 15:57:53

    nuxtjs中单独引入Message组件的问题

    // 引入elementUIimport { Message } from 'element-ui';//由于Message组件并没有install 方法供Vue来操作的,是直接返回的,因此按照官方文档单独引入的方法是//会报错的,需要给 Message 添加 install 方法Message.install = function (Vue, options) {Vue.prototype.$message = Message}Vue.use(Message )//消息提示

  • 2020-03-16 16:03:20

    css的var()函数

     随着sass,less预编译的流行,css也随即推出了变量定义var函数。var()函数,就如同sass和less等预编译软件一样,可以定义变量并且进行对应的使用。

  • 2020-03-16 16:52:05

    对icomoon的误解,以及最快速的使用

    此时需要注意顶部第一个选项,Quick Usage,一定要打开,Enable Quick Usage,谁让咱英语不好呢,这个时候会出现一个css连接,直接引用就好了,就可以随意使用图标了,引入这一个css就能实现我们的功能,省区引入太多文件的烦恼,你可以在浏览器打开这个css,可以看到里面把我们所用的文件整成base64了。所以挺好用的。

  • 2020-03-17 09:47:05

    video标签视频不自动播放的问题

    添加 muted 属性,就可以通过地址栏进入网页的时候自动播放了,手机端还是有的有限制的,比如iphone浏览器,就不行,苹果手机为了保护用户的流量和用户的意愿,是禁止自动播放的,必须有手动触发。