侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

快速入门Laravel模型事件

Laravel  /  管理员 发布于 8年前   414

Laravel 模型事件允许你监听模型生命周期内的多个关键点,甚至可以在阻止一个模型的保存或者删除。 Laravel 模型事件文档 概述了如何使用钩子将对应事件与相关的事件类型关联起来,但是本文的主旨是事件与监听器的构建与设置,并额外补充一些细节的说明。

事件概述

Eloquent 有很多事件可以让你使用钩子将它们关联起来,并且增加自定义的功能到你的模型中。该模型起始时有以下事件:

retrieved

creating

created

updating

updated

saving

saved

deleting

deleted

restoring

restored

从文档这里我们可以了解它们都是如何实现的,你还可以进入 Model 的基类去看看它们到底是如何实现的:

当现有模型被数据库检索时, retrieved 事件将会触发。当一个新的模型被第一次保存时, creating 和 created 事件将会触发。如果对一个已经存在于数据库的模型调用 save 方法, updating / updated 事件将会触发。无论怎样,在这两种情况下, saving / saved 事件都会触发。

文档中对模型事件进行了很好的概述,同时解释了怎样使用钩子去关联事件,但是如果你是初学者,或者并不是熟悉怎样使用钩子将事件监听器与这些自定义模型事件相关联,请进一步阅读本文。

注册 事件

为了在你的模型中关联一个事件,你需要做的第一件事是使用 $dispatchesEvents 属性去注册事件对象,这最终将通过 HasEvents::fireCustomModelEvent() 方法触发,该方法将通过 fireModelEvent() 方法被调用。 fireCustomModelEvent() 方法原始的时候大致是下面这样:

/** * 为给定的事件触发一个自定义模型。 * * @param  string  $event * @param  string  $method * @return mixed|null */protected function fireCustomModelEvent($event, $method){    if (! isset($this->dispatchesEvents[$event])) {        return;    }    $result = static::$dispatcher->$method(new $this->dispatchesEvents[$event]($this));    if (! is_null($result)) {        return $result;    }}

一些事件,比如 delete, 将进行检测判断是否这个事件会返回 false 然后退出操作。比如,你可以使用这个钩子去做一些检测,也可以防止一个用户被创建或删除。

使用 App\User 模型举例,这里展示了如何配置你的模型事件:

protected $dispatchesEvents = [    'saving' => \App\Events\UserSaving::class,];

你可以使用 artisan make:event 命令来为你创建这个事件,但基本上这将是你最后得到结果 :

<?phpnamespace App\Events;use App\User;use Illuminate\Queue\SerializesModels;class UserSaving{    use SerializesModels;    public $user;    /**     *  创建一个新的事件实例     *     * @param \App\User $user     */    public function __construct(User $user)    {        $this->user = $user;    }}

我们的事件提供了一个公有的 $user 属性以便你能够在 saving 事件期间访问 User 模型实例。

为了让它工作起来下一步需要做的是为这个事件建立一个实际的监听器。我们设置好模型的触发时机,当 User 模型触发 saving 事件,监听器就会被调。

创建一个事件监听器

现在,我们定义 User 模型并注册一个事件监听器来监听 saving 事件的触发。虽然,我能通过模型观察器快速实现,但是,我想引导你为单个事件触发配置事件监听器。

事件监听器就像 Laravel 其它事件监听一样,handle() 方法将接收 App\Events\UserSaving 事件类的一个实例。

你可以手动创建它,也可以使用 php artisan make:listener 命令。 不管怎么样,你都将创建一个像下面这样子监听类:

<?phpnamespace App\Listeners;use App\Events\UserSaving as UserSavingEvent;class UserSaving{    /**     * 处理事件。     *     * @param  \App\Events\UserSavingEvent $event     * @return mixed     */    public function handle(UserSavingEvent $event)    {        app('log')->info($event->user);    }}

我只是添加了一个日志记录调用,以便于检查传递给监听器的模型。为此,我们还需要在 EventServiceProvider::$listen 属性中注册监听器:

<?phpnamespace App\Providers;use Illuminate\Support\Facades\Event;use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;class EventServiceProvider extends ServiceProvider{    /**     * 应用的事件监听器。     *      * @var array     */    protected $listen = [        \App\Events\UserSaving::class => [\App\Listeners\UserSaving::class,        ],    ];    // ...}

现在,当模型调用 saving 事件时,我们注册的事件监听器也会被触发并执行。

尝试事件监听

我们可以通过 tinker 会话快速生成事件监听代码:

php artisan tinker>>> factory(\App\User::class)->create();=> App\User {#794     name: "Aiden Cremin",     email: "[email protected]",     updated_at: "2018-03-15 03:57:18",     created_at: "2018-03-15 03:57:18",     id: 2,   }

如果你已正确注册了事件和监听器,则应该在 laravel.log 文件中可以看到该模型的 JSON 表达形式:

[2018-03-15 03:57:18] local.INFO: {"name":"Aiden Cremin","email":"[email protected]"}

要注意的一点,此时模型并没有 created_at 或 updated_at 属性。如果在模型上再次调用 save() ,日志上将会有一个带有时间戳的新记录,因为 saving 事件会在新创建的记录或现在有记录上触发:

>>> $u = factory(\App\User::class)->create();=> App\User {#741     name: "Eloisa Hirthe",     email: "[email protected]",     updated_at: "2018-03-15 03:59:37",     created_at: "2018-03-15 03:59:37",     id: 3,   }>>> $u->save();=> true>>>

停止一个保存操作

某些模型事件是允许你进行阻止操作的。举个荒谬的例子,假设我们不允许任何一个用户的模型保存其属性 $user->name 的内容为 Paul :

/** * 处理事件。 * * @param  \App\Events\UserSaving $event * @return mixed */public function handle(UserSaving $event){    if (stripos($event->user->name, 'paul') !== false) {        return false;    }}

在 Eloquent 的 Model::save() 方法中,会根据事件监听的返回结果判断是否进行停止保存操作:

public function save(array $options = []){    $query = $this->newQueryWithoutScopes();    // 如果 "saving" 事件返回 false ,我们将退出保存并返回    // false,表示保存失败。这为服务监听者提供了一个机会,    // 当验证失败或者出现其它任何情况,都可以取消保存操作。    if ($this->fireModelEvent('saving') === false) {        return false;    }

这个 save() 是个很好的例子,它告诉了你如何在模型生命周期中自定义事件,以及被动执行日志数据记录或者任务调度。

使用观察者

如果你正在监听多个事件,那么你可能会发现使用观察者类来按类型分组存放事件会更加方便。这里是一个例子 Eloquent 观察者 :

<?phpnamespace App\Observers;use App\User;class UserObserver{    /**     * 监听 User 创建事件。     *     * @param  \App\User  $user     * @return void     */    public function created(User $user)    {        //    }    /**     * 监听 User 删除事件。     *     * @param  \App\User  $user     * @return void     */    public function deleting(User $user)    {        //    }}

你可以在服务提供者 AppServiceProvider 中的 boot() 方法里注册观察者。

/** * 运行所有应用服务。 * * @return void */public function boot(){    User::observe(UserObserver::class);}

推荐教程:《Laravel教程》

以上就是快速入门Laravel模型事件的详细内容,更多请关注其它相关文章!


  • 上一条:
    Laravel的Auth模块使用
    下一条:
    分享几个 Laravel 7 中很酷的 Blade 组件
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • Laravel 11.14版本发布 - 新的字符串助手和ServeCommand改进(0个评论)
    • Laravel 11.12版本发布 - Artisan的`make`命令自动剪切`.php `扩展(0个评论)
    • Laravel的轻量型购物车扩展包:binafy/laravel-cart(0个评论)
    • Laravel 11.11版本发布 - 查看模型中的第三方关系:show(0个评论)
    • 近期文章
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-07
    • 2017-08
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2020-10
    • 2020-11
    • 2021-01
    • 2021-02
    • 2021-03
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-11
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-03
    • 2022-04
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-02
    • 2024-03
    • 2024-04
    • 2024-05
    • 2024-06
    • 2024-07
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客