前言
前段時間做項目時候,想要在不改變方法簽名的情況下,給 Model::find 方法做個緩存。而且想要做到即插即用。下面話不多說了,來一起看看詳細的介紹哦。
1.先看一下當(dāng)我們調(diào)用 find 方法時,框架干了什么?
找到 Illuminate\Database\Eloquent\Model 的代碼,搜索 find,沒有該方法。看來是走了 __callStatic 這個魔術(shù)方法。該方法里只有一行代碼:
return (new static)->$method(...$parameters);
static 指的是調(diào)用該靜態(tài)方法的類(如果使用的是 UserModel::find(1),則 static 就代表 UserModel 類)。看來是實例化了一個對象,并調(diào)用了成員方法。
2.分析如何優(yōu)雅地在中間插一腳
為了能夠在調(diào)用 find 時候,先走我們的緩存,所以我們需要覆蓋 __callStatic 方法,并檢測如果是 find 方法,則優(yōu)先返回緩存中的數(shù)據(jù)。
另外,為了能夠達到即插即用的效果,我們使用繼承的方式,而是使用了 Trait。核心邏輯如下:
public static function create($data = null){
if ($data == null){
return null;
}
$instance = new static;
foreach ($data as $key => $value){
$instance[$key] = $value;
}
return $instance;
}
/**
* 如果方法是 find($id, $nocache)
*
* @param string $method
* @param array $parameters
* @return mixed
*/
public static function __callStatic($method, $parameters)
{
if ($method == 'find'){
// 從緩存中獲取數(shù)據(jù)
$obj = static::create(json_decode(Redis::get(static::getCacheKey($parameters[0])), true));
if (null == $obj){
$obj = (new static)->$method(...$parameters);
if (null == $obj){
return null;
} else {
$key = static::getCacheKey($parameters[0]);
// 設(shè)置緩存及過期時間
Redis::set($key, $obj);
Redis::expire($key, static::$expire_time);
return $obj;
}
} else {
$obj->exists = true;
return $obj;
}
} else if($method == 'findNoCache'){
$method = 'find';
return (new static)->$method(...$parameters);
}
return (new static)->$method(...$parameters);
}
private static function getCacheKey($id){
$name = str_replace('\\', ':', __CLASS__);
return "{$name}:{$id}";
}
大體邏輯上面已經(jīng)介紹過了:覆蓋 __callStatic 方法,判斷如果是調(diào)用 find ,則走緩存(無緩存,查詢后需要設(shè)置緩存)。另新增 findNoCache 方法。
3.細節(jié)補充
當(dāng)修改(或刪除)數(shù)據(jù)(調(diào)用 save 方法)時需要刪除已緩存的內(nèi)容。
private static function clearCache($id){
Redis::del(self::getCacheKey($id));
}
/**
* when save, should clear cache
* @param array $options
*/
public function save(array $options = []){
static::clearCache($this[$this->primaryKey]);
return parent::save($options);
}
// delete 方法我暫時寫,內(nèi)容類似 save 方法
如何使用。在需要使用 find 緩存的 Model 類里,加上一行就夠了。
class User extends BaseModel
{
use MemoryCacheTrait;
}
快去試試吧。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:- 在 Laravel 6 中緩存數(shù)據(jù)庫查詢結(jié)果的方法
- laravel清除視圖緩存的代碼
- 在Laravel的Model層做數(shù)據(jù)緩存的實現(xiàn)
- laravel使用Redis實現(xiàn)網(wǎng)站緩存讀取的方法詳解
- 清除laravel緩存命令代碼實例