如何利用 macro 方法来扩展 Laravel 的基础类的功能

在一般编程中,我们要扩展一个基础类,我们需要进行继承才能扩充。然而 Laravel 利用 PHP 的特性,编写了一套叫做 Macroable 的 Traits,这样,凡是使用 Macroable 的类,都是可以使用这个方法扩充的。

基本使用方法#

我下面用 Collection 类来做示范。

要给 Collection 加一个扩充方法可以这样写:

  <?php
  Collection::macro("macro_name", function ($parameters) {
      // Your macro
  });

这样就很容易的就扩充了 Collection 的方法,而不需要进行复杂的继承。

我们再举个具体的例子,把所有 Collection 的字符数组全部变成大写。那么我们就这样写:

<?php
Collection::macro('uppercase', function () {
    return collect($this->items)->map(function ($item) {
        return strtoupper($item);
    });
});

collect(["hello", "world"])->uppercase();

这个结果是: ["HELLO", "WORLD"]

关于 macro 内部的 $this#

Collection $this 在 macro 的作用域必须注意,$this 不是指向你文件类的对象,而是指向你 marco 扩充的类。比如例子中的 $this 是指向 Collection 的。

这是因为在 Marcoable 的源代码中,是可以看到 static::$macros[$method]->bindTo($this, static::class) 这段代码。而 bindTo 是改变 $this 上下文指向的方法。

marco 的代码应该放在哪里?#

marco 的代码应该放在哪里才能让整个项目都能使用, 这个问题其实困扰了我很久, 所以一直没有写这个教程。不过现在研究明白了。

要让 marco 扩充的类,保证整个项目都能使用, 需要创建一个 ServiceProvider,并把扩充的方法,放入 boot() 的方法中

  <?php
  namespace App\Providers;

  use Collection;
  use Illuminate\Support\ServiceProvider;

  class CollectionMacroServiceProvider extends ServiceProvider {

      public function boot()
      {
         Collection::macro('uppercase', function () {
            return collect($this->items)->map(function ($item) {
                return strtoupper($item);
            });
        });
      }

  }

然后我们就可以在 config/app.php 中的 providers 中下面加 App\ProvidersCollectionMacroServiceProvider::class 即可

哪些类可以使用 marco#

  • Response
  • Request
  • Collection
  • HTML
  • Form
  • Filesystem
  • Cache
  • Str
  • Arr
  • Translator

等等,使用了 Marcoable 的 Traits,如果是自己编写的类,使用了 Marcoable,也可以这样扩充使用(写 Laravel 开源库的时候)

英文简版和参考#

我把经验的部分写在了 Stackoverflow 的 document 中了,有兴趣可以支持一下:
http://stackoverflow.com/documentation/lar...

参考内容:

本帖已被设为精华帖!
本帖由系统于 7年前 自动加精
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
《L02 从零构建论坛系统》
以构建论坛项目 LaraBBS 为线索,展开对 Laravel 框架的全面学习。应用程序架构思路贴近 Laravel 框架的设计哲学。
讨论数量: 4
OMGZui

thanks,最近研究源码看到了这个宏 trait,一直没明白,现在大概懂了

7年前 评论

今天也看我们项目这样写,有点不知道所云,但是写法又觉得不错。项目是这样写的

AppServiceProvider.php

public function boot()
    {
        // 请求流水号
        $GLOBALS['_REQUEST_SN'] = date('Ymd') . '-' . Str::orderedUuid();
        // 响应宏定义
        Response::macro('output', function (string $message, $code = 0, $data = null) {
            $elapsed = round(microtime(true) - LARAVEL_START, 3);

            return response()->json([
                'code'      => $code,
                'message'   => $message,
                'data'      => $data,
                'elapsed'   => $elapsed,
                'req_sn' => $GLOBALS['_REQUEST_SN'],
            ], 200, [], JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
        });
    }

### 在控制器中这样使用


    public function success($data = [], string $message = 'success')
    {
        // 宏定义
        /** @see AppServiceProvider::boot() */
        return response()->output($message, 0, $data);
    }
4年前 评论

就是 代码提示 有点难受

3年前 评论