詳解Laravel模型事件和模型事件在Trait中的使用
模型事件
Laravel 模型事件允許你監(jiān)聽模型生命周期內(nèi)的事件, 并且通過這個(gè)事件去做一些模型通用性的東西, 例如檢查用戶修改了那個(gè)字段, 將字段的什么值修改成另外的什么值, 等等.
事件類型
# 當(dāng)現(xiàn)有模型被數(shù)據(jù)庫檢索時(shí) retrieved # 當(dāng)一個(gè)新的模型被第一次保存時(shí) creating created # 對一個(gè)已經(jīng)存在于數(shù)據(jù)庫的模型調(diào)用 save 方法 updating updated # 當(dāng)模型數(shù)據(jù)被刪除時(shí) deleting deleted # 當(dāng)創(chuàng)建和更新執(zhí)行時(shí)都會調(diào)用 saving saved # 當(dāng)啟用軟刪除的數(shù)據(jù)被恢復(fù)時(shí) restoring restored # 當(dāng)啟用軟刪除的數(shù)據(jù)被強(qiáng)制刪除時(shí) forceDeleted # 復(fù)制 replicating # 指定模型的指定事件被觸發(fā)時(shí)(注意 $name 前有個(gè)空格) # event 對應(yīng)以上的事件類型 # name 對應(yīng)的是模型的類名稱 eloquent.{$event}: {$name}
以上事件類型的注冊可以通過 boot 方法注冊
boot 方法和事件注冊
模型在執(zhí)行時(shí)候僅僅執(zhí)行一次的 boot 方法, 可以理解為模型啟動時(shí)候的注冊鉤子方法我們看下示例
<?php Class PamAccount extend Model { public static function boot() { parent::boot(); self::deleting(function ($user){ if (!method_exists((new static::class), 'bootSoftDeletes')) { $user->roles()->sync([]); } return true; }); } }
這里注冊了一個(gè)事件, 這個(gè)事件的目的是在刪除的時(shí)候(如果不是軟刪除) 刪除角色信息, 這里注冊的是一個(gè) deleting
方法, 使用的方式是 self::deleting($cb)
這個(gè)函數(shù), 除此之外其他的事件也都有相對應(yīng)的方法.
boot 在 traits 中的使用
假如有一個(gè)場景我們需要再多個(gè)模型中均需要注冊同樣的操作, 我們是否需要再多個(gè)模型中編寫重復(fù)的代碼呢 ? 答案是不需要, 因?yàn)?Trait
也提供了相關(guān)的方法來進(jìn)行注冊/初始化, 在 Model.php
中可以發(fā)現(xiàn)如下定義
protected static function boot() { static::bootTraits(); }
這里的意思是在運(yùn)行 boot
方法的時(shí)候同時(shí)注冊相關(guān) Traits
方法, 這也是 Eloquent ORM 的優(yōu)勢之一, 當(dāng)使用 Eloquent 時(shí),可以執(zhí)行 Trait 注冊邏輯。
boot
方法非常_神奇_,因?yàn)榭梢詫?trait 附加到模型,如果在 trait 上設(shè)置一些方法,這些方法將在開始使用模型時(shí)調(diào)用。它們遵循如下格式 : boot{Trait}
和 initialize{Trait}
,允許在多個(gè)模型中重用相同的代碼。
boot(啟動) 和 initialize(初始化)
兩者的區(qū)別很簡單:boot
靜態(tài)執(zhí)行 ,而 initialize
動態(tài)執(zhí)行。 boot
影響模型的所有實(shí)例,而 initialize
將僅對其實(shí)例起作用。
為了更好地舉例說明這些是如何工作的,讓我們創(chuàng)建一個(gè)名為HasToken
的 trait. 這個(gè) trait 在給定的 Eloquent 模型上有兩個(gè)任務(wù)。
- 創(chuàng)建模型時(shí)添加事件以記錄誰插入了記錄
- 將隨機(jī)字符串作為標(biāo)記添加到模型中
第一項(xiàng)任務(wù)是了解經(jīng)過身份驗(yàn)證的用戶創(chuàng)建了新模型。第二個(gè)將允許模型自動創(chuàng)建一個(gè)隨機(jī)令牌,以避免在我們應(yīng)用程序的多個(gè)位置手動創(chuàng)建它。
我們可以使用單個(gè) trait 來做到這一點(diǎn):
<?php namespace App; use Illuminate\Support\Facades\Log; use Illuminate\Support\Facades\Auth; use Illuminate\Support\Str; trait HasToken { /** * Boot the trait * * @return void */ protected function bootHasToken() { static::created(function ($model) { // Log who created this model Log::info('Token for ' . class_basename($model) . ' created by ' . Auth::user()->getKey()); }); } /** * Initialize the trait * * @return void */ protected function initializeHasToken() { // Automatically create a random token $this->token = Str::random(100); } }
Boot(啟動)
boot 方法將在靜態(tài)模型上工作。例如,如果將此 trait 添加到 Authentication
模型中,則使用此方法所做的一切都會以靜態(tài)的方式影響該模型 > 方法名稱必須遵循該boot{TraitName}
格式
這對于向 eloquent 事件添加回調(diào)非常方便,例如creating
或retrieving
,因?yàn)檫@些事件是靜態(tài)的。
/** * Boot the trait * * @return void */ protected function bootHasToken() { static::created(function ($model) { // Log who created this model Log::info('Token for ' . class_basename($model) . ' created by ' . Auth::user()->getKey() ); }); }
這是一個(gè)非常棒的想法:啟動模型不會執(zhí)行兩次,僅在需要時(shí)進(jìn)行。無需擔(dān)心 bootHasToken()
多次調(diào)用該方法,Eloquent ORM 模型會持續(xù)跟蹤啟動的模型
Initialize (初始化)
我們可以使用 boot 方法做靜態(tài)級的事情,但是要操作模型實(shí)例本身,我們需要使用初始化方法??梢允褂迷?code>initialize{Trait}格式創(chuàng)建初始化程序。
/** * Initialize the trait * * @return void */ protected function initializeHasToken() { // Automatically create a random token $this->token = Str::random(100); }
每次實(shí)例化新模型時(shí)都會運(yùn)行初始化方法. And that’s the magic. Happy coding.
參考 Laravel: Booting and Initializing Models with traits
以上就是詳解Laravel模型事件和模型事件在Trait中的使用的詳細(xì)內(nèi)容,更多關(guān)于Laravel模型事件Trait的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
PHPCMS手機(jī)站偽靜態(tài)設(shè)置詳細(xì)教程
這篇文章主要介紹了PHPCMS手機(jī)站偽靜態(tài)設(shè)置教程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-02-02PHP?ceil()函數(shù)浮點(diǎn)數(shù)向上取整實(shí)現(xiàn)示例
這篇文章主要為大家介紹了PHP?ceil()函數(shù)實(shí)現(xiàn)浮點(diǎn)數(shù)向上取整示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01php使用ICQ網(wǎng)關(guān)發(fā)送手機(jī)短信
php發(fā)送手機(jī)短信實(shí)例代碼。2013-10-10基于thinkphp6.0的success、error實(shí)現(xiàn)方法
這篇文章主要介紹了基于thinkphp6.0的success、error實(shí)現(xiàn)方法,本文分步驟給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-11-11