在Laravel的Model層做數(shù)據(jù)緩存的實(shí)現(xiàn)
您在此之前可能就已經(jīng)緩存過模型數(shù)據(jù),但是我將向您展示一個(gè)使用動(dòng)態(tài)記錄模型的更精細(xì)的Laravel模型緩存技術(shù),這是我一開始在 RailsCasts學(xué)習(xí)到的技術(shù)。
使用模型的唯一緩存鍵,您可以緩存模型(或關(guān)聯(lián)模型)更新時(shí)自動(dòng)更新(以及緩存失效)的模型上的屬性和關(guān)聯(lián),一個(gè)好處是訪問緩存的數(shù)據(jù)比在控制器中緩存的數(shù)據(jù)更具可復(fù)用性,因?yàn)樗谀P蜕隙皇窃趩蝹€(gè)控制器方法中。
這是這個(gè)技術(shù)的要點(diǎn):
假設(shè)你有很多個(gè) Comment
的 Article
模型,給定下面的Laravel blade 模板,你就可以像下面這樣訪問 /article/:id
路由時(shí)得到評(píng)論的數(shù)量:
<h3>$article->comments->count() {{ str_plural('Comment', $article->comments->count())</h3>
您可以在控制器中緩存評(píng)論的計(jì)數(shù),但是當(dāng)您有多個(gè)需要緩存的一次性查詢和數(shù)據(jù)時(shí),控制器會(huì)變得非常臃腫難看。使用控制器,訪問緩存的數(shù)據(jù)也不是很方便。
我們可以構(gòu)建一個(gè)模板,它僅在文章更新時(shí)訪問數(shù)據(jù)庫,并且訪問該模型的所有代碼都可以獲取緩存值:
<h3>$article->cached_comments_count {{ str_plural('Comment', $article->cached_comments_count)</h3>
通過使用模型訪問器,我們可以緩存基于最后一次文章更新的評(píng)論計(jì)數(shù)值。
因此,在評(píng)論新增或刪除時(shí)我們?cè)撛趺锤挛恼碌?nbsp;updated_at
列值呢?
先進(jìn)入 touch 方法看看。
模型的觸發(fā)
可以通過使用模型的 touch()
方法來更新文章的 updated_at
列值:
$ php artisan tinker >>> $article = \App\Article::first(); => App\Article {#746 id: 1, title: "Hello World", body: "The Body", created_at: "2018-01-11 05:16:51", updated_at: "2018-01-11 05:51:07", } >>> $article->updated_at->timestamp => 1515649867 >>> $article->touch(); => true >>> $article->updated_at->timestamp => 1515650910
我們可以用更新的 timestamp 值使緩存失效。不過在新增或刪除一個(gè)評(píng)論時(shí),我們?cè)趺从|發(fā)修改文章的 updated_at
字段呢?
碰巧 Eloquent 模型中有一個(gè)屬性就叫 $touches
。下面是我們的評(píng)論模型的大概樣子:
<?php namespace App; use App\Article; use Illuminate\Database\Eloquent\Model; class Comment extends Model { protected $guarded = []; protected $touches = ['article']; public function article() { return $this->belongsTo(Article::class); } }
這里的 $touches
屬性是個(gè)數(shù)組,包含了在評(píng)論的創(chuàng)建、保存和刪除時(shí)會(huì)引起“觸發(fā)”的關(guān)聯(lián)信息。
緩存的屬性
我們先回到 $article->cached_comments_count
訪問器。該方法的實(shí)現(xiàn)可能象 App\Article
模型中的樣子:
public function getCachedCommentsCountAttribute() { return Cache::remember($this->cacheKey() . ':comments_count', 15, function () { return $this->comments->count(); }); }
我們使用唯一鍵值的 cacheKey()
方法緩存模型 15 分鐘,然后簡單地在閉包方法中返回評(píng)論計(jì)數(shù)值。
注意,我們也用到了 Cache::rememberForever()
方法,靠著緩存機(jī)制的垃圾回收策略以刪除過期的鍵值。我設(shè)置了一個(gè)定時(shí)器,以便在每隔 15 分鐘的緩存刷新間隔里,緩存可在該時(shí)間的多數(shù)范圍內(nèi)有最高的命中率。
cacheKey()
方法要用到模型的唯一鍵值,并且在模型更新時(shí)對(duì)應(yīng)緩存失效。下面是我的 cacheKey
實(shí)現(xiàn)代碼:
public function cacheKey() { return sprintf( "%s/%s-%s", $this->getTable(), $this->getKey(), $this->updated_at->timestamp ); }
模型的 cacheKey()
方法示例輸出結(jié)果可能返回下面的字串信息:
articles/1-1515650910
這個(gè)鍵值是由表名、模型id值及當(dāng)前 updated_at
的 timestamp 值組成。一旦我們觸發(fā)這個(gè)模型,timestamp 值就會(huì)更新,并且我們的模型緩存就會(huì)相應(yīng)地失效。
以下是 Article
模型的完整代碼:
<?php namespace App; use App\Comment; use Illuminate\Support\Facades\Cache; use Illuminate\Database\Eloquent\Model; class Article extends Model { public function cacheKey() { return sprintf( "%s/%s-%s", $this->getTable(), $this->getKey(), $this->updated_at->timestamp ); } public function comments() { return $this->hasMany(Comment::class); } public function getCachedCommentsCountAttribute() { return Cache::remember($this->cacheKey() . ':comments_count', 15, function () { return $this->comments->count(); }); } }
然后是關(guān)聯(lián)的 Comment
模型:
<?php namespace App; use App\Article; use Illuminate\Database\Eloquent\Model; class Comment extends Model { protected $guarded = []; protected $touches = ['article']; public function article() { return $this->belongsTo(Article::class); } }
接下來做什么?
我已經(jīng)向你展示了如何緩存一個(gè)簡單的評(píng)論計(jì)數(shù),但是如何緩存所有的評(píng)論呢?
public function getCachedCommentsAttribute() { return Cache::remember($this->cacheKey() . ':comments', 15, function () { return $this->comments; }); }
你也可以選擇將評(píng)論轉(zhuǎn)換為數(shù)組替代序列化模型,只允許在前端對(duì)數(shù)據(jù)進(jìn)行簡單的數(shù)組訪問:
public function getCachedCommentsAttribute() { return Cache::remember($this->cacheKey() . ':comments', 15, function () { return $this->comments->toArray(); }); }
最后, 我在 Article
模型中定義了cacheKey()
方法,但是你可能想要通過一個(gè)名為 ProvidesModelCacheKey
的trait來定義這個(gè)方法以便你可以在復(fù)合模型中使用或者在一個(gè)基礎(chǔ)模型中定義所有模型擴(kuò)展的方法。 你甚至可能想要為實(shí)現(xiàn)cacheKey()
方法的模型使用使用契約(接口)。
我希望你已經(jīng)發(fā)現(xiàn)這個(gè)簡單的技術(shù)是十分有用的!
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
windows中為php安裝mongodb與memcache
這篇文章主要介紹了windows中為php安裝mongodb與memcache的方法,十分的詳盡,需要的朋友可以參考下2015-01-01PHP中實(shí)現(xiàn)中文字串截取無亂碼的解決方法
這篇文章主要介紹了PHP中實(shí)現(xiàn)中文字串截取無亂碼的解決方法,直接使用PHP函數(shù)substr截取中文字符可能會(huì)出現(xiàn)亂碼,下面跟隨腳本之家小編一起看看具體解決方法吧2018-05-05PHP 表單提交及處理表單數(shù)據(jù)詳解及實(shí)例
本文章向大家介紹表單的一些知識(shí)點(diǎn),然后介紹PHP是如何接收表單數(shù)據(jù)并如何處理表單數(shù)據(jù),文章以一個(gè)發(fā)送郵件的表單實(shí)例來向大家講解表單提交及php如何處理表單數(shù)據(jù),需要的朋友可以參考下2016-12-12分享一段php獲取linux服務(wù)器狀態(tài)的代碼
正在做的項(xiàng)目中,需要使用php獲取Linux服務(wù)器狀態(tài),度娘了一下,發(fā)現(xiàn)絕大部分都是基于WIN系統(tǒng)的,好吧,自力更生,分享給大家一段Linux的代碼2014-05-05PHP網(wǎng)頁游戲?qū)W習(xí)之Xnova(ogame)源碼解讀(十五)
這篇文章主要介紹了PHP網(wǎng)頁游戲Xnova(ogame)源碼解讀的艦隊(duì)活動(dòng)頁面處理流程,需要的朋友可以參考下2014-06-06Thinkphp5框架實(shí)現(xiàn)獲取數(shù)據(jù)庫數(shù)據(jù)到視圖的方法
這篇文章主要介紹了Thinkphp5框架實(shí)現(xiàn)獲取數(shù)據(jù)庫數(shù)據(jù)到視圖的方法,涉及thinkPHP5數(shù)據(jù)庫配置、讀取、模型操作及視圖調(diào)用相關(guān)操作技巧,需要的朋友可以參考下2019-08-08