Laravel - MySQL数据库的使用详解9(Eloquent ORM用法6:事件、订阅、观察者)

作者: hgphp 发布时间: 2019-09-11 浏览: 2806 次 编辑

一、事件的监听与响应

1,基本介绍

(1)Eloquent 模型可以在模型生命周期中的各个时间点触发相应的事件:

  • retrieved:从数据库中获取已存在模型时会触发该事件。
  • creatingcreated:当一个新模型被首次保存的时候,这两个事件会被触发。
  • updatingupdated:当一个模型已经在数据库中存在并调用 save 方法,这两个事件会被触发。
  • savingsaved:无论是创建还是更新,这两个事件都会被触发。

(2)我们可以通过事件实现:在一个指定模型类每次保存或更新的时候执行相应的代码。

2,使用样例

(1)首先我们打开 app/Providers/EventServiceProvider.php 文件,注册一个事件监听器映射关系:

<?php
namespace App\Providers;
use Illuminate\Support\Facades\Event;
use Illuminate\Auth\Events\Registered;
use Illuminate\Auth\Listeners\SendEmailVerificationNotification;
use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;
class EventServiceProvider extends ServiceProvider
{
    /**
     * The event listener mappings for the application.
     *
     * @var array
     */
    protected $listen = [
        Registered::class => [
            SendEmailVerificationNotification::class,
        ],
        &#39;App\Events\UserSaved&#39; => [
          &#39;App\Listeners\UserSavedListener&#39;,
        ],
    ];
    /**
     * Register any events for your application.
     *
     * @return void
     */
    public function boot()
    {
        parent::boot();
        //
    }
}

(2)接着在终端中进入项目文件夹,然后执行如下 Artisan 命令:

php artisan event:generate

(3)上面命令执行后,会分别自动在 app/Eventsapp/Listensers 目录下生成对应的事件类和监听类。

原文:Laravel - MySQL数据库的使用详解9(Eloquent ORM用法6:事件、订阅、观察者)


(4)修改生成的 UserSaved.php 文件内容:

<?php
namespace App\Events;
use Illuminate\Broadcasting\Channel;
use Illuminate\Queue\SerializesModels;
use Illuminate\Broadcasting\PrivateChannel;
use Illuminate\Broadcasting\PresenceChannel;
use Illuminate\Foundation\Events\Dispatchable;
use Illuminate\Broadcasting\InteractsWithSockets;
use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
use App\Models\User;
class UserSaved
{
    use Dispatchable, InteractsWithSockets, SerializesModels;
    public $user;
    /**
     * Create a new event instance.
     *
     * @return void
     */
    public function __construct(User $user)
    {
        $this->user = $user;
    }
    /**
     * Get the channels the event should broadcast on.
     *
     * @return \Illuminate\Broadcasting\Channel|array
     */
    public function broadcastOn()
    {
        return new PrivateChannel(&#39;channel-name&#39;);
    }
}

(5)修改生成的 UserSavedListener.php 文件内容:

<?php
namespace App\Listeners;
use App\Events\UserSaved;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue;
class UserSavedListener
{
    /**
     * Create the event listener.
     *
     * @return void
     */
    public function __construct()
    {
        //
    }
    /**
     * Handle the event.
     *
     * @param  UserSaved  $event
     * @return void
     */
    public function handle(UserSaved $event)
    {
        echo "--- saved事件响应,保存对象如下 ----\n";
        $user = $event->user;
        echo(json_encode($user));
    }
}

(6)修改模型 User.php,在模型保存时发出 UserSaved 这个自定义事件。

<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Notifications\Notifiable;
use App\Events\UserSaved;
class User extends Model {
  use Notifiable;
  /**
  * The event map for the model.
  *
  * @var array
  */
  protected $dispatchesEvents = [
    &#39;saved&#39; => UserSaved::class,
    //&#39;deleted&#39; => UserDeleted::class,
  ];
  public $timestamps = false;
}


(7)下面进行测试,我们获取一个用户,修改其属性后保存。

$user = User::find(1);
$User->age = 44;
$user->save();

(8)运行结果如下:可以看到事件监听响应成功。

原文:Laravel - MySQL数据库的使用详解9(Eloquent ORM用法6:事件、订阅、观察者)


二、事件的订阅

1,基本介绍

(1)事件订阅(Event Subscribers)是一种特殊的 Listener,前面讲的是一个 listener 里只能放一个 hander()

(2)事件订阅可以把很多处理器(handler)放到一个类里面,然后用一个 listner 把它们集合起来,这样不同的事件只要对应一个 listner 就可以了。

2,使用样例

(1)假设我们 User 模型这边在保存和删除时会分别发出 UserSavedUserDeleted 这两个自定义事件(这两个自定义事件内容我就不写了,具体可参考上面的文章内容)。

<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Notifications\Notifiable;
use App\Events\UserSaved;
class User extends Model {
  use Notifiable;
  /**
  * The event map for the model.
  *
  * @var array
  */
  protected $dispatchesEvents = [
    &#39;saved&#39; => UserSaved::class,
    &#39;deleted&#39; => UserDeleted::class,
  ];
  public $timestamps = false;
}

(2)接着我们定义一个同时处理这两个事件的 Listener

<?php
namespace App\Listeners;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue;
class UserEventListener
{
    // 用户保存事件响应
    public function onUserSaved($event) {
      echo "--- onUserSaved ----\n";
      $user = $event->user;
      echo(json_encode($user));
    }
    // 用户删除事件响应
    public function onUserDeleted($event) {
      echo "--- onUserDeleted ----\n";
      $user = $event->user;
      echo(json_encode($user));
    }
    // 同时订阅多个事件
    public function subscribe($events)
    {
      $events->listen(
        &#39;App\Events\UserSaved&#39;,
        &#39;App\Listeners\UserEventListener@onUserSaved&#39;
      );
      $events->listen(
        &#39;App\Events\UserDeleted&#39;,
        &#39;App\Listeners\UserEventListener@onUserDeleted&#39;
      );
    }
}

(3)最后在 EventServiceProvider.php 注册下就可以了:

<?php
namespace App\Providers;
use Illuminate\Support\Facades\Event;
use Illuminate\Auth\Events\Registered;
use Illuminate\Auth\Listeners\SendEmailVerificationNotification;
use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;
class EventServiceProvider extends ServiceProvider
{
    /**
     * The event listener mappings for the application.
     *
     * @var array
     */
    protected $listen = [
        Registered::class => [
            SendEmailVerificationNotification::class,
        ]
    ];
    // 注册订阅
    protected $subscribe = [
      &#39;App\Listeners\UserEventListener&#39;,
    ];
    /**
     * Register any events for your application.
     *
     * @return void
     */
    public function boot()
    {
        parent::boot();
        //
    }
}

三、观察者(Observers)

1,基本介绍

如果需要监听模型中的多个事件,我们还可以使用观察者。观察者类可以反射我们想要监听的 Eloquent 事件对应的方法名,并且每个方法接收模型作为唯一参数。

2,使用样例

(1)使用观察者的话,模型这边就不需要再指定发出哪些事件了:

<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
class User extends Model {
}

(2)接着我们创建一个User模型对应的观察者类(UserObserver)。观察者类的方法即为事件名,这样事件就可以通过反射自动找到对应的方法。

<?php
namespace App\Observers;
use App\Models\User;
class UserObserver
{
    /**
     * 监听用户创建事件.
     *
     * @param  User  $user
     * @return void
     */
    public function saved(User $user)
    {
      echo "--- onUserSaved ----\n";
      echo(json_encode($user));
    }
    /**
     * 监听用户删除事件.
     *
     * @param  User  $user
     * @return void
     */
    public function deleted(User $user)
    {
      echo "--- onUserDeleted ----\n";
      echo(json_encode($user));
    }
}

(3)最后在 app/Providers/AppServiceProvider.php 中注册观察者即可:

<?php
namespace App\Providers;
use Illuminate\Support\ServiceProvider;
use App\Models\User;
use App\Observers\UserObserver;
class AppServiceProvider extends ServiceProvider
{
    /**
     * Register any application services.
     *
     * @return void
     */
    public function register()
    {
        //
    }
    /**
     * Bootstrap any application services.
     *
     * @return void
     */
    public function boot()
    {
        User::observe(UserObserver::class);
    }
}



Laravel - MySQL数据库的使用详解系列:

1,Laravel - MySQL数据库的使用详解1(安装配置、基本用法) 4,Laravel - MySQL数据库的使用详解4(Eloquent ORM用法1:创建模型) 7,Laravel - MySQL数据库的使用详解7(Eloquent ORM用法4:插入、更新数据)
2,Laravel - MySQL数据库的使用详解2(Query Builder用法1:查询操作) 5,Laravel - MySQL数据库的使用详解5(Eloquent ORM用法2:基本查询、动态范围) 8,Laravel - MySQL数据库的使用详解8(Eloquent ORM用法5:删除数据)
3,Laravel - MySQL数据库的使用详解3(Query Builder用法2:新增、修改、删除) 6,Laravel - MySQL数据库的使用详解6(Eloquent ORM用法3:模型关联、关联查询 9,Laravel - MySQL数据库的使用详解9(Eloquent ORM用法6:事件、订阅、观察者)