溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Laravel模型事件的示例分析

發(fā)布時間:2021-03-17 10:10:35 來源:億速云 閱讀:183 作者:小新 欄目:編程語言

這篇文章主要介紹Laravel模型事件的示例分析,文中介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們一定要看完!

Laravel 模型事件允許你監(jiān)聽模型生命周期內(nèi)的多個關(guān)鍵點,甚至可以在阻止一個模型的保存或者刪除。 Laravel 模型事件文檔 概述了如何使用鉤子將對應(yīng)事件與相關(guān)的事件類型關(guān)聯(lián)起來,但是本文的主旨是事件與監(jiān)聽器的構(gòu)建與設(shè)置,并額外補充一些細(xì)節(jié)的說明。

事件概述

Eloquent 有很多事件可以讓你使用鉤子將它們關(guān)聯(lián)起來,并且增加自定義的功能到你的模型中。該模型起始時有以下事件:

retrieved

creating

created

updating

updated

saving

saved

deleting

deleted

restoring

restored

從文檔這里我們可以了解它們都是如何實現(xiàn)的,你還可以進入 Model 的基類去看看它們到底是如何實現(xiàn)的:

當(dāng)現(xiàn)有模型被數(shù)據(jù)庫檢索時, retrieved 事件將會觸發(fā)。當(dāng)一個新的模型被第一次保存時, creating 和 created 事件將會觸發(fā)。如果對一個已經(jīng)存在于數(shù)據(jù)庫的模型調(diào)用 save 方法, updating / updated 事件將會觸發(fā)。無論怎樣,在這兩種情況下, saving / saved 事件都會觸發(fā)。

文檔中對模型事件進行了很好的概述,同時解釋了怎樣使用鉤子去關(guān)聯(lián)事件,但是如果你是初學(xué)者,或者并不是熟悉怎樣使用鉤子將事件監(jiān)聽器與這些自定義模型事件相關(guān)聯(lián),請進一步閱讀本文。

注冊 事件

為了在你的模型中關(guān)聯(lián)一個事件,你需要做的第一件事是使用 $dispatchesEvents 屬性去注冊事件對象,這最終將通過  HasEvents::fireCustomModelEvent() 方法觸發(fā),該方法將通過  fireModelEvent() 方法被調(diào)用。 fireCustomModelEvent() 方法原始的時候大致是下面這樣:

/**
 * 為給定的事件觸發(fā)一個自定義模型。
 *
 * @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 然后退出操作。比如,你可以使用這個鉤子去做一些檢測,也可以防止一個用戶被創(chuàng)建或刪除。

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

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

你可以使用 artisan make:event 命令來為你創(chuàng)建這個事件,但基本上這將是你最后得到結(jié)果 :

<?php
namespace App\Events;
use App\User;
use Illuminate\Queue\SerializesModels;
class UserSaving
{
    use SerializesModels;
    public $user;
    /**
     *  創(chuàng)建一個新的事件實例
     *
     * @param \App\User $user
     */
    public function __construct(User $user)
    {
        $this->user = $user;
    }
}

我們的事件提供了一個公有的 $user 屬性以便你能夠在 saving 事件期間訪問 User 模型實例。

為了讓它工作起來下一步需要做的是為這個事件建立一個實際的監(jiān)聽器。我們設(shè)置好模型的觸發(fā)時機,當(dāng) User 模型觸發(fā) saving 事件,監(jiān)聽器就會被調(diào)。

創(chuàng)建一個事件監(jiān)聽器

現(xiàn)在,我們定義 User 模型并注冊一個事件監(jiān)聽器來監(jiān)聽 saving 事件的觸發(fā)。雖然,我能通過模型觀察器快速實現(xiàn),但是,我想引導(dǎo)你為單個事件觸發(fā)配置事件監(jiān)聽器。

事件監(jiān)聽器就像 Laravel 其它事件監(jiān)聽一樣,handle() 方法將接收 App\Events\UserSaving 事件類的一個實例。

你可以手動創(chuàng)建它,也可以使用 php artisan make:listener 命令。 不管怎么樣,你都將創(chuàng)建一個像下面這樣子監(jiān)聽類:

<?php
namespace 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);
    }
}

我只是添加了一個日志記錄調(diào)用,以便于檢查傳遞給監(jiān)聽器的模型。為此,我們還需要在 EventServiceProvider::$listen 屬性中注冊監(jiān)聽器:

<?php
namespace App\Providers;
use Illuminate\Support\Facades\Event;
use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;
class EventServiceProvider extends ServiceProvider
{
    /**
     * 應(yīng)用的事件監(jiān)聽器。
     * 
     * @var array
     */
    protected $listen = [
        \App\Events\UserSaving::class => [
            \App\Listeners\UserSaving::class,
        ],
    ];
    // ...
}

現(xiàn)在,當(dāng)模型調(diào)用 saving 事件時,我們注冊的事件監(jiān)聽器也會被觸發(fā)并執(zhí)行。

嘗試事件監(jiān)聽

我們可以通過 tinker 會話快速生成事件監(jiān)聽代碼:

php artisan tinker
>>> factory(\App\User::class)->create();
=> App\User {#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,
   }

如果你已正確注冊了事件和監(jiān)聽器,則應(yīng)該在  laravel.log 文件中可以看到該模型的 JSON 表達形式:

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

要注意的一點,此時模型并沒有 created_at 或 updated_at 屬性。如果在模型上再次調(diào)用 save() ,日志上將會有一個帶有時間戳的新記錄,因為 saving 事件會在新創(chuàng)建的記錄或現(xiàn)在有記錄上觸發(fā):

>>> $u = factory(\App\User::class)->create();
=> App\User {#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
>>>

停止一個保存操作

某些模型事件是允許你進行阻止操作的。舉個荒謬的例子,假設(shè)我們不允許任何一個用戶的模型保存其屬性 $user->name  的內(nèi)容為 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() 方法中,會根據(jù)事件監(jiān)聽的返回結(jié)果判斷是否進行停止保存操作:

public function save(array $options = [])
{
    $query = $this->newQueryWithoutScopes();
    // 如果 "saving" 事件返回 false ,我們將退出保存并返回
    // false,表示保存失敗。這為服務(wù)監(jiān)聽者提供了一個機會,
    // 當(dāng)驗證失敗或者出現(xiàn)其它任何情況,都可以取消保存操作。
    if ($this->fireModelEvent('saving') === false) {
        return false;
    }

這個  save()  是個很好的例子,它告訴了你如何在模型生命周期中自定義事件,以及被動執(zhí)行日志數(shù)據(jù)記錄或者任務(wù)調(diào)度。

使用觀察者

如果你正在監(jiān)聽多個事件,那么你可能會發(fā)現(xiàn)使用觀察者類來按類型分組存放事件會更加方便。這里是一個例子  Eloquent 觀察者 :

<?php
namespace App\Observers;
use App\User;
class UserObserver
{
    /**
     * 監(jiān)聽 User 創(chuàng)建事件。
     *
     * @param  \App\User  $user
     * @return void
     */
    public function created(User $user)
    {
        //
    }
    /**
     * 監(jiān)聽 User 刪除事件。
     *
     * @param  \App\User  $user
     * @return void
     */
    public function deleting(User $user)
    {
        //
    }
}

你可以在服務(wù)提供者 AppServiceProvider 中的 boot() 方法里注冊觀察者。

/**
 * 運行所有應(yīng)用服務(wù)。
 *
 * @return void
 */
public function boot()
{
    User::observe(UserObserver::class);
}

以上是“Laravel模型事件的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI