### 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