摘要:我們設(shè)置好模型的觸發(fā)時(shí)機(jī),當(dāng)模型觸發(fā)事件,監(jiān)聽器就會(huì)被調(diào)。雖然,我能通過模型觀察器快速實(shí)現(xiàn),但是,我想引導(dǎo)你為單個(gè)事件觸發(fā)配置事件監(jiān)聽器?,F(xiàn)在,當(dāng)模型調(diào)用事件時(shí),我們注冊的事件監(jiān)聽器也會(huì)被觸發(fā)并執(zhí)行。
Laravel 模型事件允許你監(jiān)聽模型生命周期內(nèi)的多個(gè)關(guān)鍵點(diǎn),甚至可以在阻止一個(gè)模型的保存或者刪除。 Laravel 模型事件文檔 概述了如何使用鉤子將對應(yīng)事件與相關(guān)的事件類型關(guān)聯(lián)起來,但是本文的主旨是事件與監(jiān)聽器的構(gòu)建與設(shè)置,并額外補(bǔ)充一些細(xì)節(jié)的說明。
事件概述Eloquent 有很多事件可以讓你使用鉤子將它們關(guān)聯(lián)起來,并且增加自定義的功能到你的模型中。該模型起始時(shí)有以下事件:
retrieved
creating
created
updating
updated
saving
saved
deleting
deleted
restoring
restored
從文檔這里我們可以了解它們都是如何實(shí)現(xiàn)的,你還可以進(jìn)入 Model 的基類去看看它們到底是如何實(shí)現(xiàn)的:
當(dāng)現(xiàn)有模型被數(shù)據(jù)庫檢索時(shí), retrieved 事件將會(huì)觸發(fā)。當(dāng)一個(gè)新的模型被第一次保存時(shí), ?creating?和?created?事件將會(huì)觸發(fā)。如果對一個(gè)已經(jīng)存在于數(shù)據(jù)庫的模型調(diào)用?save 方法,?updating?/?updated?事件將會(huì)觸發(fā)。無論怎樣,在這兩種情況下,?saving?/?saved?事件都會(huì)觸發(fā)。
文檔中對模型事件進(jìn)行了很好的概述,同時(shí)解釋了怎樣使用鉤子去關(guān)聯(lián)事件,但是如果你是初學(xué)者,或者并不是熟悉怎樣使用鉤子將事件監(jiān)聽器與這些自定義模型事件相關(guān)聯(lián),請進(jìn)一步閱讀本文。
注冊 事件為了在你的模型中關(guān)聯(lián)一個(gè)事件,你需要做的第一件事是使用?$dispatchesEvents?屬性去注冊事件對象,這最終將通過 ?HasEvents::fireCustomModelEvent()?方法觸發(fā), 該方法將通過 ?fireModelEvent()?方法被調(diào)用。?fireCustomModelEvent() 方法原始的時(shí)候大致是下面這樣:
/** * 為給定的事件觸發(fā)一個(gè)自定義模型。 * * @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, 將進(jìn)行檢測判斷是否這個(gè)事件會(huì)返回 false?然后退出操作。比如,你可以使用這個(gè)鉤子去做一些檢測,也可以防止一個(gè)用戶被創(chuàng)建或刪除。
使用 ?AppUser?模型舉例, 這里展示了如何配置你的模型事件:
protected $dispatchesEvents = [ "saving" => AppEventsUserSaving::class, ];
你可以使用?artisan make:event?命令來為你創(chuàng)建這個(gè)事件, 但基本上這將是你最后得到結(jié)果 :
user = $user; } }
我們的事件提供了一個(gè)公有的?$user?屬性以便你能夠在 saving事件期間訪問 User模型實(shí)例。
為了讓它工作起來下一步需要做的是為這個(gè)事件建立一個(gè)實(shí)際的監(jiān)聽器。我們設(shè)置好模型的觸發(fā)時(shí)機(jī),當(dāng)?User模型觸發(fā)?saving?事件,監(jiān)聽器就會(huì)被調(diào)。
創(chuàng)建一個(gè)事件監(jiān)聽器現(xiàn)在,我們定義 User 模型并注冊一個(gè)事件監(jiān)聽器來監(jiān)聽?saving 事件的觸發(fā)。雖然,我能通過模型觀察器快速實(shí)現(xiàn),但是,我想引導(dǎo)你為單個(gè)事件觸發(fā)配置事件監(jiān)聽器。
事件監(jiān)聽器就像 Laravel 其它事件監(jiān)聽一樣,handle() 方法將接收 AppEventsUserSaving 事件類的一個(gè)實(shí)例。
你可以手動(dòng)創(chuàng)建它,也可以使用 php artisan make:listener 命令。 不管怎么樣,你都將創(chuàng)建一個(gè)像下面這樣子監(jiān)聽類:
info($event->user); } }
我只是添加了一個(gè)日志記錄調(diào)用,以便于檢查傳遞給監(jiān)聽器的模型。為此,我們還需要在 EventServiceProvider::$listen 屬性中注冊監(jiān)聽器:
[ AppListenersUserSaving::class, ], ]; // ... }
現(xiàn)在,當(dāng)模型調(diào)用 saving 事件時(shí),我們注冊的事件監(jiān)聽器也會(huì)被觸發(fā)并執(zhí)行。
嘗試事件監(jiān)聽我們可以通過 tinker?會(huì)話快速生成事件監(jiān)聽代碼:
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, }
如果你已正確注冊了事件和監(jiān)聽器,則應(yīng)該在 ?laravel.log 文件中可以看到該模型的 JSON 表達(dá)形式:
[2018-03-15 03:57:18] local.INFO: {"name":"Aiden Cremin","email":"josie05@example.com"}
要注意的一點(diǎn),此時(shí)模型并沒有 created_at?或?updated_at 屬性。如果在模型上再次調(diào)用 save() ,日志上將會(huì)有一個(gè)帶有時(shí)間戳的新記錄,因?yàn)?saving 事件會(huì)在新創(chuàng)建的記錄或現(xiàn)在有記錄上觸發(fā):
>>> $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 >>>停止一個(gè)保存操作
某些模型事件是允許你進(jìn)行阻止操作的。舉個(gè)荒謬的例子,假設(shè)我們不允許任何一個(gè)用戶的模型保存其屬性 $user->name? 的內(nèi)容為 Paul :
/** * 處理事件。 * * @param AppEventsUserSaving $event * @return mixed */ public function handle(UserSaving $event) { if (stripos($event->user->name, "paul") !== false) { return false; } }
在 Eloquent 的 Model::save() 方法中,會(huì)根據(jù)事件監(jiān)聽的返回結(jié)果判斷是否進(jìn)行停止保存操作:
public function save(array $options = []) { $query = $this->newQueryWithoutScopes(); // 如果 "saving" 事件返回 false ,我們將退出保存并返回 // false,表示保存失敗。這為服務(wù)監(jiān)聽者提供了一個(gè)機(jī)會(huì), // 當(dāng)驗(yàn)證失敗或者出現(xiàn)其它任何情況,都可以取消保存操作。 if ($this->fireModelEvent("saving") === false) { return false; }
這個(gè) ?save()? 是個(gè)很好的例子,它告訴了你如何在模型生命周期中自定義事件,以及被動(dòng)執(zhí)行日志數(shù)據(jù)記錄或者任務(wù)調(diào)度。
使用觀察者如果你正在監(jiān)聽多個(gè)事件,那么你可能會(huì)發(fā)現(xiàn)使用觀察者類來按類型分組存放事件會(huì)更加方便。這里是一個(gè)例子? Eloquent 觀察者 :
你可以在服務(wù)提供者 AppServiceProvider 中的 boot() 方法里注冊觀察者。
/** * 運(yùn)行所有應(yīng)用服務(wù)。 * * @return void */ public function boot() { User::observe(UserObserver::class); }了解更多我建議你閱讀? Laravel 事件文檔?去了解有關(guān)事件和監(jiān)聽器在整個(gè)框架中如何工作。 Eloquent 事件文檔 ? 對于可用事件以及如何使用觀察者是一個(gè)非常好的參考。最后,我建議瀏覽 IlluminateDatabaseEloquentModel?類來查找?fireModelEvent()?方法調(diào)用的用法去了解事件如何與模型和?HasEvents?trait 將這些事件聯(lián)系在一起。
現(xiàn)代化 PHP 知識(shí)日新月異,尤其是 PHP7 出來以后,歡迎加入 PHP / Laravel 知識(shí)社區(qū) 一起成長。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/28479.html
摘要:說明本文主要講述使用作為緩存加快頁面訪問速度。何不用來做緩存,等到該達(dá)到一定瀏覽頁面后再刷新下,效率也很高??勺骶彺嫦到y(tǒng)隊(duì)列系統(tǒng)。 說明:本文主要講述使用Redis作為緩存加快頁面訪問速度。同時(shí),作者會(huì)將開發(fā)過程中的一些截圖和代碼黏上去,提高閱讀效率。 備注:作者最近在學(xué)習(xí)github上別人的源碼時(shí),發(fā)現(xiàn)好多在計(jì)算一篇博客頁面訪問量view_count時(shí)都是這么做的:利用Laravel...
摘要:原文發(fā)表在我的個(gè)人網(wǎng)站系列入門教程二最適合中國人的教程本教程示例代碼見大家在任何地方卡住,最快捷的解決方式就是去看我的示例代碼。 原文發(fā)表在我的個(gè)人網(wǎng)站:Laravel 5 系列入門教程(二)【最適合中國人的 Laravel 教程】 本教程示例代碼見:https://github.com/johnlui/Learn-Laravel-5 大家在任何地方卡住,最快捷...
摘要:過去一年時(shí)間寫了多篇文章來探討了我認(rèn)為的框架最核心部分的設(shè)計(jì)思路代碼實(shí)現(xiàn)。為了大家閱讀方便,我把這些源碼學(xué)習(xí)的文章匯總到這里。數(shù)據(jù)庫算法和數(shù)據(jù)結(jié)構(gòu)這些都是編程的內(nèi)功,只有內(nèi)功深厚了才能解決遇到的復(fù)雜問題。 過去一年時(shí)間寫了20多篇文章來探討了我認(rèn)為的Larave框架最核心部分的設(shè)計(jì)思路、代碼實(shí)現(xiàn)。通過更新文章自己在軟件設(shè)計(jì)、文字表達(dá)方面都有所提高,在剛開始決定寫Laravel源碼分析地...
閱讀 3358·2021-11-23 09:51
閱讀 1143·2021-09-03 10:30
閱讀 3275·2021-08-31 09:40
閱讀 3337·2019-08-30 14:22
閱讀 981·2019-08-30 14:09
閱讀 2967·2019-08-30 13:21
閱讀 3328·2019-08-28 18:03
閱讀 2917·2019-08-26 13:44