# 模型實戰開發之模型使用技巧
> 本節課講的模型實際上是就是常說的物理模型 (數據模型)
[TOC]
## 模型的建立
建立一個新模型很簡單 詳情參見
http://www.hmoore.net/mikkle/thinkphp5_study/268682
~~~
namespace app\base\model;
use think\Model;
/**
* Created by PhpStorm.
* User: Mikkle
* Q Q:776329498
* Date: 2017/2/8
* Time: 1:01
*/
class AdminUser extends Base
{
protected $table = "my_admin_user";
}
~~~
>[info] 模型中使用中,我們根據自身情況繼承Base基類或者子基類,在模型中
> 模型中,只要定義好$table即可正常使用了.
## TP5模型最常用且實用的方法(寫入部分)
### 模型的自動完成功能
>[success] 支持auto、insert和update三個屬性,可以分別在寫入、新增和更新的時候進行字段的自動完成機制,auto屬性自動完成包含新增和更新操作
> 模型自動完成 只需要在model中定義三種對應的數組即可,注意未賦值的數組需要定義修改器
我們現在拿那個model實例說明一下:這個是半年前寫的代碼
點開下面鏈接查看全部代碼
http://www.hmoore.net/mikkle/thinkphp5_study/329928
~~~
protected $insert = ['status'=>1,'guid','order_no','order_state'=>0,'pay_type'=>0,'send_state'=>0,'is_comment'=>0,'factory_state'=>0];
~~~
大家看一下這段代碼,我使用了 對象中$insert這個屬性賦值,負責方式是數組.
>[danger] 當定義 $insert字段時,在寫入時候模型會自動處理這些字段
> 當定義 $update字段時,在更新時候模型會自動處理這些字段
> 當定義 $auto字段時,在寫入和更新時候模型都會自動處理這些字段
其中 guid 字段和order_no字段我并沒有給出指定的值,那這時候會調用對應的修改器對該字段進行賦值
這個自動完成非常實用 特別是寫入時候的自動完成($insert).
### 模型的修改器
>[danger] 模型的修改器作用是可以在數據賦值的時候自動進行轉換處理
#### 修改器的實例
我們看一下自動完成中提到的 guid 字段和order_no字段的修改器
~~~
protected function setGuidAttr($value, $data)
{
return $this->create_uuid() ;
}
/**
* 訂單類獲取器 含防重復篩查
* Power by Mikkle
* QQ:776329498
* @param $value
* @param $data
* @return string
*/
protected function setOrderNoAttr($value, $data)
{
do {
$order_no= date('Ymd').$this->builderRand();
} while ($this->where('order_no',$order_no)->count()==1);
return $order_no ;
}
~~~
#### 模型修改器的建立方式
>[info] 修改器方法的命名規范為:setFieldNameAttr
我們只要在將字段轉換成 大寫駝峰 拼接在set. 和 Attr之間即可
| 字段名稱 | 字段大寫駝峰名稱 |拼接后修改器對應方法名稱 |
| --- | --- |--- |
| guid | Guid |setGuidAttr |
| order_no | OrderNo |setOrderNoAttr |
#### 修改器參數說明 $value, $data
| 值名稱 | 參數說明 |值類型|
| --- | --- |--- |
| $value | 當前模型中對應字段傳入的值 |--- |
| $data | 當前模型中對應字段傳入的所有值 |數組 |
#### 修改器實例講解
在訂單模型創建新數據時候,我需要在給新數據賦值一個新的GUID以及唯一的數字型訂單號,
這兩個實例都未用到$value, $data兩個參數,而使用到基類中的create_uuid builderRand
~~~
/**
* 創建個性GUID
* Power by Mikkle
* QQ:776329498
* @param string $base_code
* @return string
*/
public function create_uuid($base_code = '')
{
if (empty($base_code)) {
$base_name = basename(str_replace('\\', '/', get_called_class()), '.php');
$uuid_list = ModelCode::$uuid_list;
$base_code = isset($uuid_list[$base_name]) ? $uuid_list[$base_name] : 'QT';
}
$uuid = $base_code . strtoupper(uniqid()) . $this->builderRand(6);
return $uuid;
}
/**
* 創建隨機數
* Power by Mikkle
* QQ:776329498
* @param int $num 隨機數位數
* @return string
*/
public function builderRand($num=8){
return substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, $num);
}
~~~
>[info] 其中 create_uuid這個方法名不規范啊,現在要想改一下啊,不知道要動多少文件.
> (還有項目git中) 忍了吧 !大家千萬別學我
> 另外 setOrderNoAttr 方法中,通過while方法進行查庫比對 確保唯一
其實以前寫個的這個方法也不嚴謹,存在可能發生的bug
~~~
do {
$order_no= date('Ymd').$this->builderRand();
} while ($this->where('order_no',$order_no)->count()==1); //bug 應該 count()>0;
return $order_no ;
~~~
如果數據庫中如果已經出現重復,那查庫驗證就失效了!
好吧!我承認,我是專職寫bug的!
#### 修改器的觸發方法
修改器會在使用save方法觸發
官方文檔中也介紹了另一種觸發方法
~~~
$user = new User();
$data['name'] = 'THINKPHP';
$data['email'] = 'thinkphp@qq.com';
$user->data($data, true);
$user->save();
echo $user->name; // thinkphp
~~~
>[info] 換句話說就是使用model新增和修改都會調用和使用的修改器
### 修改器使用注意事項即只讀屬性字段的應用
那么新的問題來了,當修改數據時候,如果傳值有這兩個字段數據時候,我們發現guid和order_no字段的全部變了.而且是修改一次變更一次.這兩個字段可以說是重要索引字段 那么所有關聯數據你猜會咋樣
TP5的model方法 自讀字段的設置 就最便捷的解決方法
> 不要問我 我是如何知道這個問題的,傷疤都快沒有了,大家不要再揭了
#### 只讀字段的定義
> 只讀字段用來保護某些特殊的字段值不被更改,這個字段的值一旦寫入,就無法更改。 要使用只讀字段的功能,我們只需要在模型中定義readonly屬性:
~~~
protected $readonly = ['guid','order_no'];
~~~
#### 只讀字段放的位置
我的基類中有下面一行代碼,一般人我不告訴他 嘿嘿
~~~
protected $readonly = ['guid'];
~~~
### 自動時間戳
>[info] **protected $autoWriteTimestamp = true;**
一句話的事情 ,我每個表基本都有
### 類型轉換
這個我一直沒有怎么用到
想了解的 請查看官方手冊吧
http://www.hmoore.net/manual/thinkphp5/138669
>[info] 模型中寫入部分的常用的方法大體上 就這么多
再重申一下 一個是自動完成 一個修改器 還有一個只讀字段 這三個方法是配合使用來完成數據的數據寫入工作,
## TP5模型最常用且實用的方法(讀取部分)
### 獲取器的使用
>[info] 獲取器的作用是在獲取數據的字段值后按照我們定義的方法進行自動處理
#### 模型獲取器的建立方式
>[info] 修改器方法的命名規范為:**getFieldNameAttr**
還是看order例子中的一段代碼
~~~
public function getPayTypeAttr($value, $data){
if ($value){
$get_data = ['WxPay'=>'微信支付','AliPay'=>'支付寶支付'];
return isset($get_data[$value]) ? $get_data[$value] : '其他方式';
}else{
return $value;
}
}
public function getIsPayTextAttr($value, $data){
$get_data = ['0'=>'未付款','1'=>'已付款'];
return $get_data[$data['is_pay']];
}
~~~
我們只要在將字段轉換成 大寫駝峰 拼接在get. 和 Attr之間即可
| 字段名稱 | 字段大寫駝峰名稱 |拼接后獲取器對應方法名稱 |
| --- | --- |--- |
| pay_type | PayType |getPayTypeAttr |
| is_pay_text | IsPayText |getIsPayTextAttr |
當字段中不存在的時候,我們隨意append一個新字段,如果想獲取is_pay的中文說明,就append寫一個is_pay_text字段,并用$data中的值進行處理返回想要的值.
#### 獲取器參數說明 $value, $data
| 值名稱 | 參數說明 |值類型|
| --- | --- |--- |
| $value | 當前模型中對應字段讀取的值 |--- |
| $data | 當前模型中讀取的所有值 |數組 |
#### 獲取器的觸發方式
獲取器只有當獲取某個數據屬性的時候自動觸發,如果你要獲取包含修改器處理的全部數據屬性的話,可以使用下面的方法:
| 觸發方法 | 觸發方法實例 |
| --- | --- |
| 模型的數據對象取值操作 | $model->field_name |
| 模型的序列化輸出操作 | $model->toArray() ;$model->toJson() |
| 顯式調用getAttr方法 | $this->getAttr('field_name') |
| 附加字段要append | `$model->append(['field_name'])->toArray()` |
備注:定義修改器的字段 如果要獲取原始數據的時候 使用下面官方文檔中的方法
~~~
$user = User::get(1);
// 通過獲取器獲取字段
echo $user->status;
// 獲取原始字段數據
echo $user->getData('status');
// 獲取全部原始數據
dump($user->getData());
~~~
#### 獲取器小結
>[danger] 獲取器在項目中使用率比較高,我個人不建設使用和字段同名的獲取器,而使用別名的獲取器.
> 特別是項目進行中時候,你添加的同名獲取器,可能會引發老代碼bug
- 序言及更新日志
- 前言一 開發PHP必備的環境(你可以不看)
- LinUX系統ThinkPHP5鏈接MsSQL數據庫的pdo_dblib擴展
- centos7.2掛載硬盤攻略
- Centos系統Redis安裝及Redis的PHP擴展安裝
- Centos系統增加Swap(系統交換區)的方法
- 前言二 開發PHP軟件配置和介紹(你依然可以不看)
- 數據庫SQL文件
- 本地Git(版本控制)的搭建
- GIT遠程倉庫的克隆和推送
- Git常用命令
- PHP面向對象思想實戰經驗領悟
- PHP面向對象實戰----命名空間
- PHP面向對象實戰----繼承
- 基類實戰--底層方法封裝
- 基類實戰--構造函數實戰
- 基類實戰--析構函數的使用
- TP5實戰開發前篇---控制器(controller)
- 控制器中Request類的使用
- 控制器中基類的使用
- TP5實戰開發前篇---模型篇(model)
- TP5實戰開發前篇---驗證器篇(Validate)
- TP5實戰課程入門篇---花拳繡腿
- 模塊以及類的文件的建立
- Api開發------單條信息顯示
- Api開發---單條信息復雜關聯顯示
- Api開發---查詢信息緩存Cache的應用
- TP5實戰技巧---開發思路 引路造橋
- TP5實戰技巧---整合基類 化繁為簡
- TP5實戰課程入門篇---數據操作
- Api開發---數據的添加和修改
- API開發---快速開發API通用接口
- TP5專用微信sdk使用教程
- THINKPHP5微信SDK更新記錄及升級指導
- TP5專用SDK 微信參數配置方法
- 微信公眾號推送接口對接教程
- 微信推送接口對接示例含掃描登錄微信端部分
- TP5專用微信支付SDK使用簡介
- TP5專用支付寶支付SDK使用說明
- 使用NW將開發的網站打包成桌面應用
- TP5高階實戰課程 進階篇概述
- 進階篇一 實戰開發之習慣及要求
- 進階篇二 實戰開發之控制器
- 控制器基類之控制器基類使用方法
- 控制器基類之控制器基類常用方法分享
- 控制器基類之構造函數的使用方法
- 進階篇三 實戰開發之權限控制
- TP5實戰源碼 --- 全局用戶信息驗證類Auth
- TP5實戰源碼 --- 微信Auth實戰開發源碼
- 進階篇四 實戰開發之模型
- 模型基類之模型基類的用途
- 模型基類之常用數據處理方法
- 模型邏輯層之實戰代碼(含事務)
- 模型實戰開發之模型常用方法
- 模型實戰源碼 --- 樂觀鎖的應用
- 模型實戰技巧---Model事件功能的使用
- 模型事件實戰應用---數據庫操作日志
- 進階篇五 實戰開發之緩存(Cache)
- TP5實戰源碼---應用緩存獲取城市信息
- TP5實戰源碼---應用緩存獲取分類詳情
- 進階篇六 TP5類庫的封裝和使用
- DataEdit快捷操作類庫
- ShowCode快捷使用類庫
- 阿里大于 短信API接口 TP5專用類庫
- DatabaseUpgrade數據庫對比及更新類庫
- AuthWeb權限類使用說明
- 進階篇七 服務層的應用
- 服務層源碼示例
- 服務層基類源碼
- 進階篇八 應用層Redis數據處理基類
- Redis服務層基類源碼
- 進階篇九 使用Redis類庫處理一般的搶購(秒殺)活動示例
- 進階篇十 某大型項目應用本Redis類源碼示例(含事務 樂觀鎖)
- 進階篇十一 邏輯層的應用
- 邏輯層基類源碼
- 進階篇 服務層代碼示例
- 高階實戰課程 進階篇持續新增中
- 高階篇一 TP5命令行之守護任務源碼
- TP5實戰源碼 --- 命令行
- TP5實戰源碼 --- 通過shell建立PHP守護程序
- 高階篇二 使用Redis隊列發送微信模版消息
- 高階篇二 之 Worker隊列基類源碼
- 高階篇三 TP5實戰之Redis緩存應用
- Redis實戰源碼之Hash專用類庫源碼
- Redis實戰源碼之Model類結合
- Redis實戰源碼之模型Hash基類源碼
- Redis實戰源碼之Hash查詢使用技巧
- Redis實戰源碼之 shell腳本中redis賦值和取值
- 高階篇四 Swoole的實戰應用
- swoole基類代碼
- Swoole擴展WebsocketServer專用類
- 基于Swoole的多Room聊天室的程序
- Swoole守護服務shell源碼
- 高階篇五 命令行異步多進程隊列類的應用
- tp_worker類源碼
- WorkerBase
- WorkerCommand
- WorkerRedis
- Redis類
- CycleWorkBase
- WorkerHookBase異步鉤子
- 隊列日志SQL
- 高階篇六 定時執行隊列類庫以及使用方法
- 定時隊列類庫源碼
- 高階篇七 異步執行循環隊列類庫以及使用教程
- CycleWorkBase源碼
- 高階實戰課程 進階篇持續新增中
- Extend便捷類庫源碼庫
- 阿里相關類庫
- SendSms--驗證碼API接口文件
- 權限相關類庫目錄
- AuthWeb 權限驗證類庫
- Redis便捷操作類庫(20171224更新)
- Redis
- Tools工具類庫集
- Curl類庫
- DataEdit
- Rand類庫
- ShowCode類庫
- Upload類庫
- 附件集合
- 附件一:微信支付 實戰開發源碼
- 微信支付類庫源代碼
- Common_util_pub.php
- DownloadBill_pub.php
- JsApi_pub.php
- NativeCall_pub.php
- NativeLink_pub.php
- OrderQuery_pub.php
- Refund_pub.php
- RefundQuery_pub.php
- SDKRuntimeException.php
- ShortUrl_pub.php
- UnifiedOrder_pub.php
- Wxpay_client_pub.php
- Wxpay_server_pub.php
- WxPayConf_pub.php
- 微信支付回調頁面源碼
- 附件二 順豐快遞BSP接口實戰開發源碼
- 順豐快遞BSP接口實戰開發源碼
- 順豐BSP基類
- 順豐BSP基礎代碼
- 順豐BSP下單接口
- 順豐BSP查單接口
- 順豐BSP確認/取消接口
- 附件三 APP注冊登陸接口源碼(含融云平臺接口)
- 附件四 TP5訂單Model(含事務 獲取器 修改器等方法)
- 附錄五 RSA加密解密
- Rsa文件源碼
- 附件六 阿里大于短信接口
- 附件七 AES加解密類
- AES加解密類源碼
- 附件八 TP5路由設置源碼
- 附件九 TP5 Excel導入導出下載便捷類庫
- Excel類庫TP5源碼
- 附件十 TP5便捷操作Redis類庫源碼
- TP5源碼 Redis操作便捷類庫
- 附件十一 TP5源碼 上傳文件入庫類源碼
- 上傳類Upload源碼
- Upload類上傳配置文件
- 存儲圖像文件的數據庫SQL文件
- 存儲文件的數據庫SQL文件
- 附件十二 TP5 圖片處理增強類 支持縮略圖在線顯示
- 附件十三 微信推送消息接口類庫源碼
- 附件十三 微信推送消息接口類庫源碼 之 基類
- 附件十四 存儲微信昵稱的處理方法