### 2018 年 11 月 2 日 發布
模型三大利器我們已經了解了[修改器](https://blog.thinkphp.cn/817548)和[搜索器](https://blog.thinkphp.cn/783775),如果還不是特別理解的可以參考之前的博客。
今天來總結下獲取器的用法,其實獲取器是最容易理解的,但卻又是最容易困惑和出問題的。
## 定義獲取器
獲取器的作用是對模型對象的(原始)數據做出自動處理。一個獲取器對應模型的一個特殊方法(該方法必須為`public`類型),方法命名規范為:
>[info]### get**FieldName**Attr
`FieldName`為數據表字段的駝峰轉換或者你數據表不存在的字段(注意理解后面這句話),下面是一個典型的獲取器定義:
```
<?php
namespace app\index\model;
use think\Model;
class User extends Model
{
public function getUserTypeAttr($value, $data)
{
$type = [0 => '普通', 1 => 'VIP', 2 => '黃金', 3 => '白金', 4 => '鉆石'];
return $type[$value];
}
}
```
你需要給每一個需要輸出轉換處理的數據字段定義一個對應的獲取器,但獲取器的字段名不一定要和數據表的字段名一致,例如我希望給`user_type`字段定義一個名為`getTypeAttr`的獲取器也是允許的,但要注意這個時候傳入獲取器的第一個參數肯定是沒有值(因為沒有對應的數據表字段數據),只能通過第二個參數獲取你需要的數據。
```
<?php
namespace app\index\model;
use think\Model;
class User extends Model
{
public function getTypeAttr($value, $data)
{
$type = [0 => '普通', 1 => 'VIP', 2 => '黃金', 3 => '白金', 4 => '鉆石'];
return $type[$data['user_type']];
}
}
```
當然更為嚴謹的情況下,你還需要判斷下是否存在`$data['user_type']`,這個暫且略過。
>[danger] 注意第二個參數的`data`數據,可能本身已經經過了獲取器的處理(如果你定義了相關的獲取器的話)。
為什么要定義一個和數據報字段不一致的獲取器呢?最明顯的好處可以區分不同的字段獲取原始數據和處理過的數據。事實上,有很多理由可以讓你定義一些數據表不存在的字段獲取器,這恰恰是獲取器的魅力所在。
看的出來獲取器定義本身沒什么難度,關鍵在于方法里面的獲取邏輯,這是實際應用中最需要關注的。
## 調用獲取器
定義獲取器之后會在下列情況自動觸發:
* 模型的數據對象取值操作(例如`$model->field_name`);
* 模型的序列化輸出操作(例如`$model->toArray()`或`toJson()`);
* 顯式調用`getAttr`方法(例如`$model->getAttr('field_name')`);
前面兩種其實最終都是調用最后一種來實現的,最關鍵的是要理解第一種。模型對象取值的時候一般都是通過下面的方式:
```
$user = User::get(1);
echo $user->name;
echo $user->user_type;
```
當我們使用上面的方式進行模型對象數據獲取或者在模板輸出的時候事實上都會按照下面的順序來檢測和獲取數據。
* 第`1`步——如果查詢結果包含該字段數據,取回原始數據,否則并進入第2步;
* 第`2`步——檢查是否定義該字段的獲取器(包括動態獲取器),如果有,則調用獲取器返回結果,沒有則進入第3步;
* 第`3`步——檢查是否定義了字段的類型轉換,有則進行轉換處理并返回結果,沒有則進入第4步;
* 第`4`步——如果是系統的時間字段,則自動進行時間格式化處理并返回結果,否則進入第5步;
* 第`5`步——如果第1步檢查的時候不包含該字段數據,則檢查是否存在關聯屬性定義,有則通過關聯關系獲取數據并返回結果,否則拋出屬性未定義的異常。
>[info] 上面的這五個步驟的詳細代碼,如果你有興趣的可以直接參考`think\model\concern\Attribute`的`getAttr`方法代碼。
簡單來說,當你獲取`$user->user_type`的時候都會去檢查是否定義了相關的獲取器,而不管`user_type`字段是否是一個真實的數據表字段。
但很多情況下,你不會一個個去獲取模型數據,而是把整個模型數據返回給客戶端或者模板。
```
public function index()
{
$user = User::get(1);
return json($user);
}
```
在這種情況下,其實就是在響應輸出的時候進行了模型的`toJson`處理。
>[danger] 有一點至關重要,如果你的獲取器定義了非數據表的字段,是不會自動輸出的,必須通過`append`方法追加額外屬性(并且支持追加關聯模型屬性)。
如果我們定義了一個`type`屬性的獲取器(假設這并不是一個真實的數據表字段),那么需要使用下?的方式才能正常輸出(否則你可能只有`user_type`數據):
```
public function index()
{
$user = User::get(1);
return json($user->append(['type']));
}
```
如果你是使用`toArray`的話,處理方式相同。
如果是數據集查詢的話,一樣可以使用`append`方法統一追加額外字段。
```
public function index()
{
$users = User::all();
return json($users->append(['type']));
}
```
>[info] 除了`append`方法之外,我們還支持用`hidden`方法臨時隱藏一些數據。
## 獲取原始數據
有些情況下,除了要獲取處理過的數據外,還需要獲取原始數據以便應對不同的需求。
如果?你的獲取器都是用的區分于實際數據表字段的額外屬性字段,那么這個問題本身已經解決了。所以我們主要討論的是當你的獲取器屬性和數據表字段一致的情況下,該如何獲取原始數據。
一個最簡單的辦法是使用`getData`方法:
```
$user = User::get(1);
// 獲取user_type獲取器數據
echo $user->user_type;
// 獲取原始的user_type數據
echo $user->getData('user_type');
// 獲取全部原始數據
dump($user->getData());
```
## 動態獲取器
前面我們提到過動態獲取器的概念,動態獲取器就是不需要在模型類里面定義獲取器方法,而是在查詢的時候使用閉包來定義一個字段的獲取器對數據進行統一的處理。
```
User::withAttr('name', function($value, $data) {
return strtolower($value);
})->select();
```
如果你需要定義多個動態獲取器,多次調用`withAttr`方法就行。
動態獲取器的意義除了可以不用在模型里面定義獲取器方法之外,還可以起到覆蓋已經定義的獲取器的作用,并且動態獲取器可以支持`Db`類操作,彌補了`Db`操作不能使用獲取器的缺憾,具體就看自己的需求來選擇了。
```
Db::name('user')->withAttr('name', function($value, $data) {
return strtolower($value);
})->select();
```
## 總結
無論是獲取器,還是之前提?的修改器、搜索器,其作用無非是把你的模型工作細化和拆分,這樣代碼和邏輯也會更清晰,可維護性也大大增強,至于性能,從來不是模型首先考慮的。
- 值得升級到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