## TP5实战开发---模型技巧
在MVC模式下 model 的灵活应用可以减少代码量,逻辑处理也更加清晰明确
下面 我拿一个商城的订单Model实例来简单介绍TP5的模型 (model)的用法
~~~
<?php
/**
* Created by PhpStorm.
* User: Mikkle
* Email:776329498@qq.com
* Date: 2017/1/4
* Time: 17:00
*/
namespace app\base\model;
use think\Db;
use think\Model;
class Orders extends Base
{
protected $table = "my_orders";
protected $name = "orders";
protected $pk = "id";
protected $insert = ['status'=>1,'guid','order_no','order_state'=>0,'pay_type'=>0,'send_state'=>0,'is_comment'=>0,'factory_state'=>0];
protected $autoWriteTimestamp = true;
}
~~~
先看一下前一部分
>[info] protected $table = "my_orders"; //表名
> protected $name = "orders";
> protected $pk = "id";
> protected $insert = ['status'=>1,'guid','order_no','order_state'=>0,'pay_type'=>0,'send_state'=>0,'is_comment'=>0,'factory_state'=>0];
> protected $autoWriteTimestamp = true; //开启时间戳
其中这一部分主要说的是 自动完成 protected $insert
在这部分中后面定义了一个数组,其中有一部分已经定义值 而未赋值将自动调用修改器完成。
### Model中的获取器和修改器
>[info] 详情参见 我这里只讲实例
> 获取器 http://ihavenolimitations.xyz/manual/thinkphp5/135192
> 修改器 http://ihavenolimitations.xyz/manual/thinkphp5/135193
在这个订单表中 使用获取器
~~~
/**
* 获取器
* Power by Mikkle
* QQ:776329498
* @param $value
* @param $data
* @return mixed
*/
public function getDistributeAttr($value, $data){
$get_data = ['SF'=>'顺丰快递','ZT'=>'中通快递'];
return $get_data[$value];
}
public function getPayTimeAttr($value, $data){
return date('Y-m-d h:i:s', $value);
}
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']];
}
~~~
这几个获取器都比较简单,都是常用数据转换。
* * * * *
下面的源码是修改器
~~~
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 ;
}
~~~
其中 setGuidAttr 使用model基类的create_uuid方法创造guid,而setOrderNoAttr是生产不重复的订单号,生成方法依然使用了基类的方法。这里使用 do while循环进行了一次唯一性过滤(当数值存在,重新获取)
>[danger] 需要注意的是,为了防止订单号的重复,这里使用一次查库过滤,虽然重复的几率低,但是这种类型的值进行一次查库还是有必要的。
* * * * *
下面这个例子是是分页搜索的list列表,这里附加了字段不存在的cover_info的获取器,
~~~
/**
* 获取订单列表 含append用法
* Power by Mikkle
* QQ:776329498
* @param array $map
* @param string $order
* @param int $limit
* @param int $page
* @return array
*/
public function getOrderList($map=[],$order = 'id desc',$limit=8,$page=1){
$field='order_no,order_num,amount,distribute,create_time,is_pay,send_state,order_state';
$map['status']=1;
$data=$this->where($map)->field($field)
->limit($limit)
->page($page)
->order($order)->select();
$re=[];
foreach($data as $order){
$re[]= $order->append(['cover_info'])->toArray();
}
return $re;
}
~~~
>[danger] 注意:模型中使用select返回的是多个对象的数组,对象中的获取器只有在单个对象toArray才会全部生效,
$re=[];
foreach($data as $order){
$re[]= $order->append(['cover_info'])->toArray();
}
return $re;
所以在这个获取器最后加入foreach循环
* * * * *
下面这个例子是订单详情
~~~
/**
* 订单详情
* Power by Mikkle
* QQ:776329498
* @param array $map
* @return array|bool|false|\PDOStatement|string|Model
*/
public function getOrderInfo($map=[]){
if (!$map) return false;
$field = 'order_no,amount,distribute,order_desc,consignee_name,consignee_mobile,consignee_address,send_state,is_pay,is_comment,order_state,create_time';
$data=$this->where($map)->field($field)->find();
if (!$data) return false;
//$data['cart_list']=$data->hasManyThrough('VGuidAll','OrdersAccess','order_no','my_guid','order_no')->select();
$data['cart_list']=$this->getOrderAccessList($data['order_no']);
return $data;
}
/**
* 订单简要信息
* Power by Mikkle
* QQ:776329498
* @param $order_no
* @return array|bool|false|\PDOStatement|string|Model
*/
public function getOrderSimpleInfo($order_no){
if (!$order_no) return false;
$map['order_no']=$order_no;
$field = 'order_no,amount,distribute,order_desc,consignee_name,consignee_mobile,consignee_address,create_time';
$data=$this->where($map)->field($field)->find();
if (!$data) return false;
return $data;
}
/*
* 关联
*/
public function orderAccess(){
$this->hasMany('OrdersAccess','order_no','order_no');
}
/**
* 视图查询
* Power by Mikkle
* QQ:776329498
* @param $order_no
* @return false|\PDOStatement|string|\think\Collection
*/
public function getOrderAccessList($order_no){
return Db::view('my_orders_access','id,order_no,my_guid,my_num,my_price,is_comment')
->view('my_guid_all','my_name,my_type,my_price,my_pic','my_orders_access.my_guid=my_guid_all.my_guid')
->view('my_common_picture','path','my_common_picture.id=my_guid_all.my_pic')
->where('order_no',$order_no)
->cache('OrderList_'.$order_no)
->select();
}
~~~
>[danger] 在getOrderInfo方法中,比较复杂在初期写的时候使用了关联,后期修改会使用单独的方法getOrderAccessList
> $data['cart_list']=$this->getOrderAccessList($data['order_no']);
> 在方法getOrderAccessList中使用了视图view的方法,还使用了缓存(订单商品列表是禁止修改的,所以这里为了加快查询速度使用了缓存)
> PS:其实实现的方法很多,商品详情列表使用获取器也是可以完成的,大家也可以想一下。
* * * * *
下面这个实例是Model的数据库的关联和事务操作
~~~
try{
$this->startTrans();
$this->data($order_data)->isUpdate(false)->save();
$new_order = $this->order_no;
if (!is_numeric($new_order)) throw new \Exception("商品订单添加失败");
$this->hasMany('OrdersAccess', 'order_no', 'order_no')->saveAll($data_order);
$this->commit();
return ['code' => '1001', 'msg' => '商品订单添加成功', 'data' => ['order_no' => $new_order, 'cart_list' => $data_order]];
}catch (\Exception $e){
$this->rollback();
return ['code'=>'1008','msg'=>'商品订单添加失败','data'];
}
~~~
注意:数据只要涉及多表一致性操作,必须要开启数据库事务操作
* 更多model示例请见附件4
- 序言及更新日志
- 前言一 开发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 图片处理增强类 支持缩略图在线显示
- 附件十三 微信推送消息接口类库源码
- 附件十三 微信推送消息接口类库源码 之 基类
- 附件十四 存储微信昵称的处理方法