### 2018 年 11 月 10 日 发布
对于本文涉及到的数据查询的几个基本原则请尽量纳入你的项目规范,也是官方倡导的最佳实践。在此之前,我希望你已经看过之前的一篇博客:「 [你真的了解Db类和模型的正确使用姿势么?](https://blog.thinkphp.cn/810719)」。
## 尽量不要使用数组条件查询
大部分混乱的查询语法都是使用了数组查询导致的,而`5.1`的数组条件查询用法又和`5.0`是完全不同的,如果你习惯了`5.0`的数组查询方式,建议你阅读下这篇文章:「 [教你使用5.1的数组对象查询](https://blog.thinkphp.cn/778497)」。
下面可能是很多新手比较容易犯的一个查询错误。
```
$where['id'] = ['in', '1,2,3'];
User::where($where)->select();
```
显然,这个查询思维深受老版本的影响。`5.1`版本的查询语法相比较`5.0`来说,更加对象化,下面的这种才是正确的用法。
```
$where['id'] = [1,2,3];
User::where($where)->select();
```
也许是因为PHP的数组太好用的缘故,很多人对数组查询条件乐此不疲(或者是对象焦虑?)。但如果你正确的使用查询构造器以及配合模型的相关特性,可以让你的查询逻辑变得更清晰,也更加易于维护。
而且,在一些较为复杂的查询条件下,你无法使用数组完成查询,例如下面的查询用法。
```
User::where('id', '>', 100)
->whereOr('id', '<', 10)
->where('name', 'like', 'think%')
->whereColumn('name', 'nickname')
->when('80'== $condition, function ($query) {
$query->where('score', '>', 80)->limit(10);
})->select();
```
所以,除非你很清楚`5.1`的数组查询用法,否则请尽量不要用数组条件查询了。
## 安全使用字符串查询条件
在使用字符串查询条件的时候,如果存在外部变量,请务必使用参数绑定,并最好使用`whereRaw`方法,该方法可以和其它的查询构造器方法混合使用。
```
User::whereRaw("id = :id AND name = :name", [
'id' => [$id, \PDO::PARAM_INT] ,
'name' => $name
])->where('status', 1)
->order('id', 'desc')
->select();
```
对于一些比较在意性能的查询,你也可以直接使用`query`或者`execute`方法,但同样也要注意参数的安全以及考虑不同数据库的移植问题。
```
Db::query("select * from think_user where id=? AND status=?", [8, 1]);
Db::execute("update think_user set name=:name where status=:status", ['name' => 'thinkphp', 'status' => 1]);
```
## 对使用了`SQL`函数的查询采用`Raw`机制
如果你的查询里面包含了SQL函数,那么请使用`whereRaw`(或者`whereExp`)、`orderRaw`或者`fieldRaw`方法。
```
User::whereExp('nickname', "= CONCAT(name, '-', id)")
->orderRaw("field(name,'thinkphp', 'kancloud')")
->fieldRaw('id,SUM(score)')
->select();
```
## 合理运用闭包,但不要滥用
闭包查询在查询构造器中有一些特殊用途,但如非必要,也无需滥用。
闭包查询的典型使用场景包括下面几个。
条件查询中通常都用闭包来表示一组条件查询。
```
User::when($condition, function ($query) {
// 满足条件后执行
$query->where('score', '>', 80)->limit(10);
}, function ($query) {
// 不满足条件执行
$query->where('score', '>', 60);
})->select();
```
在一些子查询中经常会用到闭包。
```
User::whereIn('id', function ($query) {
$query->table('profile')
->where('name', 'like', 'think%')
->field('id');
})->select();
```
生成一组闭合的查询条件
```
User::where('id', '>', 100)
->whereOr(function($query) {
$query->where('name', 'like', 'think%')
->whereColumn('name', 'nickname');
})->select();
```
在这个查询用法中,闭包里面的查询条件会在两边加上括号而成为一个闭合的查询条件。
在很多的关联预载入查询中可以通过闭包来进行关联数据的筛选。
```
User::with(['profile' => function($query) {
$query->field('user_id,email,phone');
}])->select([1,2,3]);
```
## 尽量复用你的查询条件
所有的查询条件应该做到一处定义多处复用,例如封装到模型的方法里面,尤其不要直接把一堆复杂的查询条件写到你的控制器代码,否则一旦业务调整,满世界的搜索代码改变你的查询条件将会是一场噩梦。
>[danger] 你也许在官方的手册或者一些教程中看到很多在控制器里面直接封装查询条件的写法,但那仅仅是出于方便展示用法的需要,并不可取。
在一些中大型的应用架构设计中,通常会把模型分成数据层、逻辑层和服务层,控制器只会调用服务层方法。而查询逻辑则基本上被封装到逻辑层里面,数据层仅仅是做模型的各种定义。
而在简单的应用里面,也可以采用PHP的`Trait`机制来实现代码的复用机制。
## 用查询范围或搜索器简化查询
如果你使用模型查询的话,把你的查询条件尽量封装到查询范围或者搜索器方法里面,查询范围和搜索器的区别主要在于查询范围比较适合定义一组(多个字段)查询条件,如果要调用多个查询范围需要多次调用,而搜索器比较适合定义一个字段(其实并非绝对)的查询条件,只需要调用一次`withSearch`方法。
使用查询范围和搜索器的例子。
```
<?php
namespace app\index\model;
use think\Model;
class User extends Model
{
public function scopeVip($query)
{
$query->where('user_type', 'vip')
->where('status', 1)
->field('id,name');
}
public function searchAgeAttr($query, $age)
{
$query->where('age','>',$age);
}
public function searchScoreAttr($query, $score)
{
$query->where('score','<=',$score)->where('score', '>' ,0);
}
}
```
控制器代码
```
<?php
namespace app\index\controller;
use think\Controller;
use think\Request;
class index extends Controller
{
public function index(Request $request)
{
// 查询VIP会员
User::vip()->select();
// 查询年龄和分数
User::withSearch(['age,'score''], $request->param())->select();
}
}
```
在控制器代码中,我们只关注业务逻辑本身,而不需要关注这个逻辑内部的查询条件是什么。更详细的关于搜索器和查询范围的内容可以参考官方手册。
>[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