资讯专栏INFORMATION COLUMN

Laravel 模型事件入门

DandJ / 3463人阅读

摘要:我们设置好模型的触发时机,当模型触发事件,监听器就会被调。虽然,我能通过模型观察器快速实现,但是,我想引导你为单个事件触发配置事件监听器。现在,当模型调用事件时,我们注册的事件监听器也会被触发并执行。

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 然后退出操作。比如,你可以使用这个钩子去做一些检测,也可以防止一个用户被创建或删除。

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

protected $dispatchesEvents = [
    "saving" => AppEventsUserSaving::class,
];

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

user = $user;
    }
}

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

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

创建一个事件监听器

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

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

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

info($event->user);
    }
}

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

 [
            AppListenersUserSaving::class,
        ],
    ];

    // ...
}

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

尝试事件监听

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

php artisan tinker
>>> factory(AppUser::class)->create();
=> AppUser {#794
     name: "Aiden Cremin",
     email: "josie05@example.com",
     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":"josie05@example.com"}

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

>>> $u = factory(AppUser::class)->create();
=> AppUser {#741
     name: "Eloisa Hirthe",
     email: "gottlieb.itzel@example.com",
     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  AppEventsUserSaving $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 观察者 :


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

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

我建议你阅读  Laravel 事件文档 去了解有关事件和监听器在整个框架中如何工作。 Eloquent 事件文档   对于可用事件以及如何使用观察者是一个非常好的参考。最后,我建议浏览 IlluminateDatabaseEloquentModel 类来查找 fireModelEvent() 方法调用的用法去了解事件如何与模型和 HasEvents trait 将这些事件联系在一起。

现代化 PHP 知识日新月异,尤其是 PHP7 出来以后,欢迎加入 PHP / Laravel 知识社区 一起成长。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28479.html

相关文章

  • Laravel学习笔记之Redis保存页面浏览量

    摘要:说明本文主要讲述使用作为缓存加快页面访问速度。何不用来做缓存,等到该达到一定浏览页面后再刷新下,效率也很高。可作缓存系统队列系统。 说明:本文主要讲述使用Redis作为缓存加快页面访问速度。同时,作者会将开发过程中的一些截图和代码黏上去,提高阅读效率。 备注:作者最近在学习github上别人的源码时,发现好多在计算一篇博客页面访问量view_count时都是这么做的:利用Laravel...

    z2xy 评论0 收藏0
  • Laravel 5 系列入门教程(二)【最适合中国人的 Laravel 教程】

    摘要:原文发表在我的个人网站系列入门教程二最适合中国人的教程本教程示例代码见大家在任何地方卡住,最快捷的解决方式就是去看我的示例代码。 原文发表在我的个人网站:Laravel 5 系列入门教程(二)【最适合中国人的 Laravel 教程】 本教程示例代码见:https://github.com/johnlui/Learn-Laravel-5 大家在任何地方卡住,最快捷...

    未东兴 评论0 收藏0
  • Laravel核心解读--完结篇

    摘要:过去一年时间写了多篇文章来探讨了我认为的框架最核心部分的设计思路代码实现。为了大家阅读方便,我把这些源码学习的文章汇总到这里。数据库算法和数据结构这些都是编程的内功,只有内功深厚了才能解决遇到的复杂问题。 过去一年时间写了20多篇文章来探讨了我认为的Larave框架最核心部分的设计思路、代码实现。通过更新文章自己在软件设计、文字表达方面都有所提高,在刚开始决定写Laravel源码分析地...

    laoLiueizo 评论0 收藏0
  • laravel入门

    摘要:开发根目录测试分为单元测试和功能测试创建一个文件执行测试测试前清除配置缓存运行单个测试用例小提示在开发与进行交互的第三方扩展包时,最好选择注入契约而不使用。 参考https://laravelacademy.org/ 概念 单词 契约Contract 就是接口 repository 仓库(封装数据访问,可以搜索:repository模式) Container 容器 ServicePr...

    韩冰 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<