Yii2 查询构建器 QueryBuilder
查询构造器 QueryBuilder
1、什么是查询构建器
- 查询构建器也是建立在 DAO 基础之上,可让你创建程序化的、DBMS 无关的 sql 语句,并且这样创建的 sql 语句比原生的
- sql 语句更易读、更安全。
$rows = (new yii\db\Query()) // 以下为构建查询
->select(['id','email']) // 查询的字段
->from('user')
->where(['last_name' => 'Carroll])
->orderBy(id)
->limit(10)
->indexBy(id) // 将 id 作为数组的键
->all(); // 这就是执行查询
2、使用查询构建器的步骤
a、构建查询。创建一个 yii\db\Query 对象来代表一条 SELECT SQL 语句,然后通过调用一套可以串起来的方法,
比如 select 方法,from 方法,where 方法等这些方法,构建出可以满足一定要求的查询条件。
(1)select() 方法, 使用字符串或者一个数组来指定需要查询的字段
$query -> select('id,email'); // 字符串形式
$query -> select(['id','email']) // 数组形式
$query -> select('user.id AS user_id, email') // 还可以是设定字段的别名形式
$query -> select(["CONTACT(first_name,' ',last_name) AS full_name",'email']) //支持 SQL 的表达式
(2) 子查询 。
SELECT id,(SELECT COUNT() FROM user) AS count FROM post
//子查询方法为:
$subQuery = (new Query()) -> select('COUNT()') -> from('user');
$query = (new Query()) -> select(['id','count' => $subQuery] -> from('post');
(3)可以调用 yii\db\Query::addSelect() 方法来选取附加字段
$query -> select(['id','username']) -> addSelect(['email']); // 如果程序需要执行到某个判断之后才能决定是否需要更多的查询字段,此时使用 addSelect()
(4) 若没有写 select() 方法,就相当于 select()
b、 from() 方法
(1)from 方法指定了 SQL 语句当中的 FROM 子句。
SELECT FROM user
$query -> from('user');
(2)from 中的表名可包含数据库前缀,以及表别名。
$query -> from(['public.user u',public.post p]); // 包含数据库前缀, 数组形式
$query -> from('public.user u,public.post p'); // 包含数据库前缀, 字符串形式
(3)可以使用子查询的结果作为表名
SELECT FROM (SELECT id FROM user WHERE status=1) u
// 子查询方法为:
$subQuery = (new Query()) -> select('id') -> from('user') -> where('status=1');
$query = (new Query()) -> from(['u'=>$subQuery]);
c、where() 方法
- 字符串格式 'status=1'
- 键值对数组格式 ['status' => 1,'type'=>2]
- 操作符格式 ['like','name','test']
d、要注意 sql 安全问题:
$query -> where('status=$status');
// 一定要写成:
$query -> where('status=:status)-> addParams([':status'=>$status]);
e、orderBy() 方法
(1) 数组形式:
$query -> orderBy([
'id' => SORT_ASC, // 升序
'name' => SORT_DESC, // 降序
])
(2)字符串形式
$query -> orderBy('id ASC,name DESC');
(3)limit() 和 offset() 方法,用来指定 SQL 语句当中的 LIMIT 和 OFFSET 子句的。
// ...... limit 10 offset 20
$query -> limit(10) -> offset(20)
(4)groupBy()
// .....groupBy(['id','status']);
$query -> groupBy(['id','status']);
可以调用 addGroupBy()来为 groupBy子句添加额外的字段
$query -> addGroupBy('age')
(5) having()
// ...... HAVING status = 1
$query -> having(['status' => 1]);
可以调用 andHaving() 或 orHaving() 方法来为 HAVING 子句追加额外的条件
// HAVING(status = 1) AND (age > 30)
$query -> having(['status' => 1]) -> andHaving(['>','age',30);
(6) join()
// ..... LEFT JOIN post ON post.user_id = user.id
$query -> join('LEFT JOIN','post','post.user_id = user.id'); // 依次为:关联类型,关联的表名,关联条件,后面还可以接可选参数 $params [与连接条件绑定的参数]
(7)union(),用来指定 SQL 语句当中的 UNION 子句的
$query1 = (new yii\db\query())
->select('id, category_id AS type,name')
->from('post')
->limit(10); $query2 = (new yii\db\query())
->select('id, type,name')
->from('user')
->limit(10); $query1 -> union($query2);
(8) indexBy 索引查询结果
当在调用 all() 方法时,它将返回一个以连续的整数值为索引的数组。而有时候希望使用一个特定的字段或者表达式的值
来作为索引结果集数组,那么在调用 yii\db\Query::all() 之前使用 yii\db\Query::indexBy() 方法来达到这个目的。
3、执行查询。执行 yii\db\Query 的一个查询方法从数据库当中检索数据。比如 all(),one(),column()等这些查询方法。
yii\db\Query 提供了一套用于不同查询目的的方法:
- all() ----------- 返回一个由行组成的数组,每一行是一个由名称和值构成的关联数组
- one() ----------- 返回结果集的第一行
- column() ----------- 返回结果集的第一列
- scalar() ----------- 返回结果集的第一行第一列的标量值
- exists() ----------- 返回一个表示该查询是否包含结果集的值
- count() ----------- 返回 COUNT 查询的结果
- sum() ----------- 返回指定列的和值
- average() ----------- 返回指定列的平均值
- max() ----------- 返回指定列的最大值
- min() ----------- 返回指定列的最小值
4、数据库查询总结
| 查询方式 | 构建查询 | 返回值(all, one 方法) |
| Command 对象 | SQL 语句 | 数组 |
| AR 的 findBySql 方法 | SQL 语句 | 对象 |
| Query 对象 | 查询构建器 | 数组 |
| 1、可程序化构建 | ||
| 2、DBMS无关 | ||
| 3、易读 | ||
| 4、更安全 | ||
| AR 的 find 方法 | 查询构建器 | 对象 |
| 1、可程序化构建 | ||
| 2、DBMS无关 | ||
| 3、易读 | ||
| 4、更安全 |
动态设置数据库连接
在controller的beforeAction里面,重新设置db组件。
/**
* @Purpose : 动态设置数据库连接
*/
public function beforeAction($action)
{
if(parent::beforeAction($action)){
\Yii::$app->set('db', [
'class' => 'yii\db\Connection',
'dsn' => 'mysql:host=' . $dbHost . ';dbname=' . $dbName,
'enableSchemaCache' => YII_DEBUG ? false : true,
'username' => $dbUser,
'password' => $dbPassword,
'charset' => 'utf8',
'emulatePrepare' => true,
'tablePrefix' => 'xx_',
]);
return true;
}
return false;
}
注:本文为作者(44106-kangaroo) 看完魏羲教你学Yii2.0 视频后所记,如有转载请注明出处:http://www.cnblogs.com/chrdai/p/8006695.html
Yii2 查询构建器 QueryBuilder的更多相关文章
- yii2 查询构建器
Query Builder $rows = (new \yii\db\Query()) ->select(['dyn_id', 'dyn_name']) ->from('zs_dynast ...
- [转]Laravel 数据库实例教程 —— 使用查询构建器实现对数据库的高级查询
本文转自:https://laravelacademy.org/post/920.html 上一节我们简单介绍了如何使用查询构建器对数据库进行基本的增删改查操作,这一节我们来探讨如何使用查询构建器实现 ...
- Laravel5.1 数据库-查询构建器
今儿个咱说说查询构建器.它比运行原生SQL要简单些,它的操作面儿也是比较广泛的. 1 查询结果 先来看看它的语法: public function getSelect() { $result = DB ...
- DB门面,查询构建器,Eloquent ORM三者的CURD
一.DB门面 1.insert DB::insert('insert into table(`name`) value(?)', ['test']); 2.update DB::update('upd ...
- Laravel 数据库实例教程 —— 使用查询构建器对数据库进行增删改查
原文地址:https://blog.csdn.net/lmy_love_/article/details/72832259 获取查询构建器很简单,还是要依赖DB门面,我们使用DB门面的table方法, ...
- laravel5.6操作数据curd写法(查询构建器)
laravel5.6 数据库操作-查询构建器 <?php //laravel5.6 语法 demo示例 namespace App\Http\Controllers;//命名该控制App空间下名 ...
- Yii2将查询数据变为键值对数组及查询构建器
- laravel 查询构建器(连贯操作)
注:laravel 查询返回行的都是 php 的 stdClass 对象实例,不是数组!!!! 1)查询多行(get) DB::table('table_name')->get(); 带偏移和限 ...
- 定义查询构建器IFeatureLayerDefinition
在宗地出图,需要实现,只显示某一户人的地块.在ArcMap里,有个定义查询,可只显示过滤后的要素. 在代码中,也比较好实现,使用IFeatureLayerDefinition接口即可. IFeatur ...
随机推荐
- GAN-生成手写数字-Keras
from keras.models import Sequential from keras.layers import Dense from keras.layers import Reshape ...
- OOP和面向对象
OOP具有三大特点 1.封装性:也称为信息隐藏,就是将一个类的使用和实现分开,只保留部分接口和方法与外部联系,或者说只公开了一些供开发人员使用的方法.于是开发人员只 需要关注这个类如何使用,而不用去关 ...
- pycaffe训练的完整组件示例
pycaffe训练的完整组件示例 为什么写这篇博客 1. 需要用到pycaffe 因为用到的开源代码基于Caffe:要维护的项目基于Caffe.基本上是用Caffe的Python接口. 2. 训练中想 ...
- office web apps搭建与解决方案
微软office在线预览解决方案https://view.officeapps.live.com/op/view.aspx?src=http://storage.xuetangx.com/public ...
- Eclipse+Maven整合开发Java项目(一)➣Maven基础环境配置
概述 Maven是一个Java语言编写的开源项目管理工具,是Apache软件基金会的顶级项目.主要用于项目构建,依赖管理,项目信息管理.有些项目需要添加响应的依赖包,Maven就是公用包集合.存在远程 ...
- Python_时间复杂度概念
时间频度:一个算法中的语句执行次数称为语句频度或时间频度,记为T(n)(T代表次数,n代表问题规模) 时间复杂度:呈现时间频度的变化规律,记为T(n)=O(f(n)) 指数时间:一个问题求解所需的执行 ...
- vsftp为不同用户设置不同的ftp的根目录
需求 要求ftp登录后的根目录是/var/test/,但是又不能影响其他用户的登录路径,因为有些程序是直接在根目录进行操作的,而没有目录切换的过程.操作过程新建用户 useradd test1user ...
- jenkins(2): jenkins定时构建项目
参考:http://blog.sina.com.cn/s/blog_b5fe6b270102v7xo.html https://blog.csdn.net/xueyingqi/article/deta ...
- 月薪3万的python程序员都看了这本书
想必大家都看过吧 Python编程从入门到实践 全书共有20章,书中的简介如下: 本书旨在让你尽快学会 Python ,以便能够编写能正确运行的程序 —— 游戏.数据可视化和 Web 应用程序,同时掌 ...
- JAVA 关键字及其作用解释
1. 访问控制 1) private 私有的 private 关键字是访问控制修饰符,可以应用于类.方法或字段(在类中声明的变量). 只能在声明 private(内部)类.方法或字段的类中引用这些类. ...