如何利用 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...

参考内容:

本帖已被设为精华帖!
本帖由系统于 6年前 自动加精
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
讨论数量: 4
OMGZui

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

5年前 评论

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

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);
    }
2年前 评论

就是 代码提示 有点难受

2年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!