### 2018 年 12 月 4 日 發布
大部分`ThinkPHP`應用都可以簡單通過緩存提升應用性能,本文就來總結下`ThinkPHP`的數據緩存使用和注意事項。
>[info] 這里說的緩存僅針對應用的數據緩存,不包含系統的緩存優化設置的使用(可以參考之前的[如何有效提高ThinkPHP的應用性能](https://blog.thinkphp.cn/843679)一文)。
`ThinkPHP5`的應用數據緩存主要包含三塊:數據緩存、查詢緩存和請求緩存。
## 數據緩存
如果你有些場景對數據的實時性要求并不高,那么選擇數據緩存是一個很好的解決辦法,數據緩存指的是直接通過`Cache`類管理緩存,例如下面是一個簡單的例子。
~~~
use think\facade\Cache;
if (Cache::has('user_data')) {
$users = Cache::get('user_data')
} else {
$users = User::where('score', '>', 100)->select();
// 緩存用戶數據30秒
Cache::set('user_data', $users, 30);
}
~~~
緩存的配置通過`config/cache.php`文件配置。
事實上,上面的查詢代碼可以簡化成:
```
$users = Cache::remember('user_data', function(){
return User::where('score', '>', 100)->select();
},30);
```
>[danger] 為了避免緩存寫入失敗的并發問題,`remember`方法默認會自動增加緩存鎖機制。
如果你希望獲取一次緩存數據后就立刻刪除該緩存,可以使用`pull`方法。
```
$users = User::where('score', '>', 100)->select();
// 緩存用戶數據30秒
Cache::set('user_data', $users, 30);
// 獲取緩存數據后刪除
$users = Cache::pull('user_data');
```
更多的緩存操作可以參考官方手冊的[緩存章節](http://www.hmoore.net/manual/thinkphp5_1/354116)。
>[info] 要發揮緩存的優勢,關鍵還在于緩存類型的合理選擇,尤其在高并發的情況下,你需要選擇分布式緩存類型。而且有時候選擇數據庫類型作為緩存機制并不會比文件緩存性能低,尤其是使用內存表的話。目前`Redis`作為緩存甚至和數據庫同步的方案已經是較為常見,網上有很多相關的文章。
## 查詢緩存
查詢緩存是專門針對數據庫查詢而設計的一種緩存簡化策略,不需要添加復雜的代碼,可以更簡單的實現數據查詢的數據緩存。代碼改造量也相對較小,只需要給你的查詢或者寫操作添加一個`cache`鏈式方法,例如上面的數據緩存代碼使用查詢緩存方式可以改成:
```
$users = User::where('score', '>', 100)
->cache('user_data', 30)
->select();
```
>[info] 查詢緩存支持所有的查詢操作,不僅是`find`,也包括`select`/`find`/`value`/`column`方法以及衍生方法。
一旦你的查詢中使用了`cache`方法并開啟了查詢緩存,系統在進行實際的數據庫查詢之前會首先檢查緩存數據是否有效,如果有效則直接返回而不再查詢,如果查詢緩存不存在或者無效則會在本次查詢后對查詢結果進行緩存。
`cache`方法其實有三個參數,最簡單是用法是只傳入一個`true`,表示開啟查詢緩存,緩存有效期則使用系統的緩存設置。
```
// 查詢緩存的有效期使用系統設置
$user = User::where('id', 10)
->cache(true)
->find();
```
如果你需要設置緩存有效期,可以使用
```
// 查詢緩存30秒有效
$user = User::where('id', 10)
->cache(30)
->find();
```
你還可以設置一個緩存失效時間
```
// 查詢緩存將在2019-1-1 失效
$expireTime = new \DateTime('2019-01-01');
$user = User::where('id', 10)
->cache($expireTime)
->find();
```
如果你需要在查詢外部操作緩存數據,可以設置一個緩存Key
```
// 設置查詢緩存的Key
$user = User::where('id', 10)
->cache('user_cache', 30)
->find();
// 在查詢外部操作緩存
$user = Cache::get('user_cache');
Cache::rm('user_cache');
```
>[danger] 推薦自己設置緩存`key`的方式使用查詢緩存,好處是可以減少緩存標識的生成開銷提高性能,而且可以在外部操作緩存。
查詢緩存目前不支持關聯模型的數據自動緩存,如果你需要緩存關聯模型的數據,需要使用數據緩存把整體的查詢結果緩存起來,可以使用下面的代碼:
```
$users = Cache::remember('users', function(){
return User::with('profile')
->where('status', 1)
->select();
},30);
```
### 緩存自動更新
緩存數據的自動更新是緩存管理的一個難點,但對于一些規范的查詢緩存,框架已經幫你實現了數據變化的時候自動更新緩存。
這里的緩存自動更新是指一旦數據更新或者刪除后會自動清理緩存(下次獲取的時候會自動重新緩存)。
自動更新規則一,當你在使用查詢緩存后,當刪除或者更新數據的時候,可以通過調用相同緩存`key`的`cache`方法,則會自動更新(清除)緩存,例如:
~~~
$users = User::cache('user_data')->select([1,3,5]);
User::cache('user_data')
->where('id', 1)
->update(['name'=>'thinkphp']);
$users = User::cache('user_data')->select([1,5]);
~~~
雖然使用了相同的緩存標識`user_data`,但最后查詢的數據不會受第一條查詢緩存的影響,最終的查詢緩存會變成第二次查詢的結果。
>[danger] 這種方式的緩存自動更新適用于所有的查詢方法,但必須確保查詢和更新或者刪除使用相同的緩存標識才能自動清除緩存。
自動更新規則二,如果使用`find`/`get`方法并且使用主鍵查詢的情況,不需要指定緩存標識,會自動清理緩存,例如:
~~~
User::cache(true)->find(1);
User::where('id', 1)->update(['name'=>'thinkphp']);
User::cache(true)->find(1);
~~~
如果是基于主鍵的查詢緩存,發生數據變化的時候會自動更新緩存,而且更新的時候無需指定`cache`方法。
## 請求緩存
數據緩存和查詢緩存只是緩存了部分數據,控制器的執行邏輯開銷依然存在,請求緩存則更進一步,可以緩存當前請求的整個響應輸出的頁面(僅對`GET`請求類型有效)。
可以在路由規則里面調用`cache`方法設置當前路由規則的請求緩存,例如:
~~~
// 定義GET請求路由規則 并設置3600秒的緩存
Route::get('new/:id','News/read')->cache(3600);
~~~
第二次訪問相同的路由地址的時候,會自動獲取請求緩存的數據響應輸出,并發送`304`狀態碼。
如果你沒有使用路由或者只定義了部分路由的話,可以通過設置全局請求緩存,并支持設置排除規則,在`app.php`文件中增加配置定義如下:
~~~
'request_cache' => true,
'request_cache_expire' => 3600,
'request_cache_except' => [
// 排除列表和user模塊
'/blog/index',
'/user',
],
~~~
排除規則里面只需要設置不使用請求緩存的地址的開頭部分(不區分大小寫)。
>[info] 路由中設置的請求緩存依然有效并且優先,如果需要設置特殊的請求緩存有效期就可以直接在路由中設置。
- 值得升級到5.1的18個理由
- 5.1.7版本新特性
- JSON字段類型在ORM中的使用
- 文件下載響應對象
- 教你使用5.1的數組對象查詢
- 模型三大利器之一:搜索器
- 在ThinkPHP中使用Yaconf
- 掌握命令行的表格輸出
- 5.1.25查詢參數綁定的改進
- ThinkPHP安全規范指引
- 巧用數據集的排序功能實現統計排序
- think-orm ——基于5.1的獨立ORM庫
- think-template——基于ThinkPHP的獨立模板引擎
- ThinkPHP5.1.26版本發布——修正版本,包含安全更新
- ThinkPHP5.0和3.2再發安全更新
- 官宣:ThinkPHP發布首個LTS版本
- 你真的了解Db類和模型的正確使用姿勢么?
- 如何更有效的記錄和管理日志
- 模型三大利器之二:修改器
- ThinkPHP5.1.28版本發布——修正上一版本問題,改進關聯查詢
- 模型三大利器之三:獲取器
- API版本控制的幾種思路
- ThinkPHP5.2第一個Beta版本發布測試
- 讓你少犯錯的數據查詢基本原則
- ThinkPHP發布5.1.29版本——常規更新
- 這15個好習慣讓你更容易升級到5.2
- 如何有效提高ThinkPHP的應用性能
- 讓你提高開發效率的查詢技巧
- 模型關聯查詢不完全指南
- 5.2發布Beta2版本——統一和精簡大量用法
- ThinkPHP發布5.1.30版本——支持微秒時間字段寫入
- ThinkPHP的數據緩存使用
- ThinkPHP5.2安裝及入口文件
- ThinkPHP榮獲2018 年度最受歡迎中國開源開發框架第1名
- 5.1路由使用心得技巧
- ThinkPHP5.*版本發布安全更新
- ThinkPHP項目及代碼規范指北
- 5.2版本的設計規范指導
- ThinkPHP5.1.32版本發布——圣誕快樂
- 利用Trait特性給模型增加樂觀鎖功能
- 5.2數據庫和模型的變化(摘要)
- ThinkPHP模板引擎實現和常見問題
- ThinkPHP5.0.24版本發布——安全更新
- 不忘初心,方得始終——ThinkPHP十三周年報告
- ThinkPHP5+相關資源匯總
- 異步社區ThinkPHP周年慶專享優惠活動
- 5.2路由的調整和改進
- ThinkPHP發布5.1.33版本——包含安全更新
- ThinkPHP擴展開發指南
- ThinkPHP發布5.2Beta3版本
- ThinkPHP發布5.1.34版本——喜迎新年
- ThinkPHP發布5.2RC1版本
- ThinkPHP發布5.1.35版本——常規更新
- 5.2配置類的調整
- 5.2時間查詢的改進和優化
- 5.2RC版本升級不完全指導(僅供學習參考)
- ThinkPHP5.2版本正式變更為6.0版本
- ThinkPHP百度云云虛擬主機專享免費活動
- 事件系統以及查詢事件、模型事件的使用
- ThinkPHP6.0RC2版本發布——架構升級、精簡核心
- ThinkPHP5.1.36LTS版本發布——常規更新
- 新版Session和Cookie設計變化
- ThinkPHP5.1.37版本發布——常規更新
- ThinkPHP6.0RC3版本發布——細節完善,體驗優化
- 6.0中間件使用詳解
- Composer各大廠商鏡像地址
- ThinkPHP6.0發布計劃公告
- 「ThinkPHP開發者周刊」招募志愿者
- ThinkPHP6.0日志變化
- ThinkPHP5.1.38版本發布——常規更新
- ThinkPHP6.0RC4版本發布——ORM獨立,日志多通道支持
- ThinkORM2.0開發指南上線
- ThinkPHP6.0RC5版本發布——多應用模式獨立,中間件機制調整
- ThinkPHP6.0版本發布——程序員節福利
- ThinkPHP5.1.39LTS版本發布——常規更新
- ThinkPHP6.0.1版本發布——圣誕快樂!
- 回顧2019,展望2020!
- ThinkPHPV6.0.2版本發布——2020新春快樂!
- 周年福利系列:Swoole合作優惠
- 億速云成為ThinkPHPV6.0獨家贊助發布商??
- 新冠疫情工具和限免資源專題(保持更新中)
- 周年福利系列:創宇信用認證合作優惠
- 周年福利系列:碼云企業版限時10%優惠
- 周年福利系列:想天短說抵現優惠
- think-swoole直播:從零開始掌握swoole開發
- 周年福利系列:B2C開源電商ShopXO授權8折優惠
- 周年福利系列:LayuiAdmin 永久授權限時優惠
- ThinkPHP資源導航站上線——構建生態 服務未來
- ThinkPHP官方技術支持服務和應用服務市場上線公測
- ThinkPHP市場精選——推廣基本要素
- ThinkPHP市場精選——客服聊天專題
- ThinkPHPV6.0.3版本發布——端午安康
- ThinkPHP開發者扶持計劃
- 6.0.3版本關鍵更新及升級事項
- 「ThinkPHP開發者周刊」改版重啟
- ThinkPHP市場精選——企業建站專題
- ThinkPHP 提供統一API接口服務
- ThinkPHP市場精選——直播電商專題
- ThinkAPI服務SDK發布
- 官方服務市場啟用獨立子域名
- ThinkPHP市場精選——刷臉支付專題
- ThinkAPI推出會員服務計劃
- ThinkPHPV6.0.4版本發布——中秋國慶雙節快樂
- ThinkPHPV5.1.40版本發布——常規更新
- 1024程序員節福利走一波
- ThinkPHP V6.0.5版本發布——兼容Composer2.0
- 知識圖譜應用場景——源論技術沙龍
- ThinkPHP5.*版本改進Composer2.0的兼容
- 官方市場雙十一精選推薦
- 技術人做產品有機會么(文末送課程)
- 本周秒殺——古德云售后獲客營銷系統
- ThinkAPI服務更新——支持接口分組和PHP版本依賴調整
- PHP8新特性盤點
- PHP8新特性系列:構造器屬性提升使用及注意事項
- ThinkPHP2021新年寄語
- ThinkPHP V6.0.6&V5.1.41版本發布——兼容PHP8.0
- PHP如何更優雅地調用API接口
- ThinkPHP V6.0.7發布——修正版本
- ThinkAPI服務更新——IP白名單
- 最新版ThinkORM對于時間字段的調整
- ThinkAPI短信接口正式上線
- ThinkPHP V6.0.8版本發布——多環境變量配置支持
- 頂想云寫作服務開啟第一次公測
- ThinkSSL上線——官方SSL/TLS證書服務
- MDBootstrap國內用戶福利——ThinkPHP官方市場首發
- ThinkPHP V6.0.9版本發布——常規更新
- ThinkORM功能盤點——虛擬模型
- 全面支持主流GIT版本庫——云寫作服務第二次公測
- 云寫作服務私有化部署方案之:版本庫私有化
- 看云雙十一活動
- ThinkPHP V6.0.10LTS發布——兼容PHP8.1
- ThinkPHP V6.0.12發布——命令行兼容8.1
- 頂想云知識管理上線公測——構建企業文檔中心和知識庫
- 頂想云上線——助力生態數字化建設
- 618活動進行中——官方市場迎來一波更新
- 頂想云知識管理正式上線——看云文檔啟動遷移服務
- ThinkPHP V6.0.13發布——常規更新
- 頂想云網站助理服務上線——構建產品支持服務
- ThinkPHP發布6.1.0&6.0.14版本——安全更新
- ThinkPHP新版社區上線試運營
- ThinkAPI上架人臉核身接口——助力網站實名認證
- 辭舊迎新——舊版社區停止注冊及發帖
- ThinkPHP6.1.2版本發布——兼容PHP8.2