Yii provides a basic database access layer as described in the Database basics section. The database access layer provides a low-level way to interact with the database. While useful in some situations, it can be tedious and error-prone to write raw SQLs. An alternative approach is to use the Query Builder. The Query Builder provides an object-oriented vehicle for generating queries to be executed.
A typical usage of the query builder looks like the following:
$rows = (new \yii\db\Query)
->select('id, name')
->from('tbl_user')
->limit(10)
->all();
// which is equivalent to the following code:
$query = (new \yii\db\Query)
->select('id, name')
->from('tbl_user')
->limit(10);
// Create a command. You can get the actual SQL using $command->sql
$command = $query->createCommand();
// Execute the command:
$rows = $command->queryAll();
As you can see, [[yii\db\Query]] is the main player that you need to deal with. Behind the scene,
Query
is actually only responsible for representing various query information. The actual query
building logic is done by [[yii\db\QueryBuilder]] when you call the createCommand()
method,
and the query execution is done by [[yii\db\Command]].
For convenience, [[yii\db\Query]] provides a set of commonly used query methods that will build the query, execute it, and return the result. For example,
- [[yii\db\Query::all()|all()]]: builds the query, executes it and returns all results as an array.
- [[yii\db\Query::one()|one()]]: returns the first row of the result.
- [[yii\db\Query::column()|column()]]: returns the first column of the result.
- [[yii\db\Query::scalar()|scalar()]]: returns the first column in the first row of the result.
- [[yii\db\Query::exists()|exists()]]: returns a value indicating whether the query results in anything.
- [[yii\db\Query::count()|count()]]: returns the result of a
COUNT
query. Other similar methods includesum()
,average()
,max()
,min()
, which support the so-called aggregational data query.
In the following, we will explain how to build various clauses in a SQL statement. For simplicity,
we use $query
to represent a [[yii\db\Query]] object.
In order to form a basic SELECT
query, you need to specify what columns to select and from what table:
$query->select('id, name')
->from('tbl_user');
Select options can be specified as a comma-separated string, as in the above, or as an array. The array syntax is especially useful when forming the selection dynamically:
$query->select(['id', 'name'])
->from('tbl_user');
Info: You should always use the array format if your
SELECT
clause contains SQL expressions. This is because a SQL expression likeCONCAT(first_name, last_name) AS full_name
may contain commas. If you list it together with other columns in a string, the expression may be split into several parts by commas, which is not what you want to see.
When specifying columns, you may include the table prefixes or column aliases, e.g., tbl_user.id
, tbl_user.id AS user_id
.
If you are using array to specify the columns, you may also use the array keys to specify the column aliases,
e.g., ['user_id' => 'tbl_user.id', 'user_name' => 'tbl_user.name']
.
To select distinct rows, you may call distinct()
, like the following:
$query->select('user_id')->distinct()->from('tbl_post');
To specify which table(s) to select data from, call from()
:
$query->select('*')->from('tbl_user');
You may specify multiple tables using a comma-separated string or an array.
Table names can contain schema prefixes (e.g. 'public.tbl_user'
) and/or table aliases (e.g. 'tbl_user u'
).
The method will automatically quote the table names unless it contains some parenthesis
(which means the table is given as a sub-query or DB expression). For example,
$query->select('u.*, p.*')->from(['tbl_user u', 'tbl_post p']);
When the tables are specified as an array, you may also use the array keys as the table aliases (if a table does not need alias, do not use a string key). For example,
$query->select('u.*, p.*')->from(['u' => 'tbl_user u', 'p' => 'tbl_post']);
You may specify a sub-query using a Query
object. In this case, the corresponding array key will be used
as the alias for the sub-query.
$subQuery = (new Query)->select('id')->from('tbl_user')->where('status=1');
$query->select('*')->from(['u' => $subQuery]);
Usually data is selected based upon certain criteria. Query Builder has some useful methods to specify these, the most powerful of which being where
. It can be used in multiple ways.
The simplest way to apply a condition is to use a string:
$query->where('status=:status', [':status' => $status]);
When using strings, make sure you're binding the query parameters, not creating a query by string concatenation. The above approach is safe to use, the following is not:
$query->where("status=$status"); // Dangerous!
Instead of binding the status value immediately, you can do so using params
or addParams
:
$query->where('status=:status');
$query->addParams([':status' => $status]);
Multiple conditions can simultaneously be set in where
using the hash format:
$query->where([
'status' => 10,
'type' => 2,
'id' => [4, 8, 15, 16, 23, 42],
]);
That code will generate the following SQL:
WHERE (`status` = 10) AND (`type` = 2) AND (`id` IN (4, 8, 15, 16, 23, 42))
NULL is a special value in databases, and is handled smartly by the Query Builder. This code:
$query->where(['status' => null]);
results in this WHERE clause:
WHERE (`status` IS NULL)
Another way to use the method is the operand format which is [operator, operand1, operand2, ...]
.
Operator can be one of the following:
and
: the operands should be concatenated together usingAND
. For example,['and', 'id=1', 'id=2']
will generateid=1 AND id=2
. If an operand is an array, it will be converted into a string using the rules described here. For example,['and', 'type=1', ['or', 'id=1', 'id=2']]
will generatetype=1 AND (id=1 OR id=2)
. The method will NOT do any quoting or escaping.or
: similar to theand
operator except that the operands are concatenated usingOR
.between
: operand 1 should be the column name, and operand 2 and 3 should be the starting and ending values of the range that the column is in. For example,['between', 'id', 1, 10]
will generateid BETWEEN 1 AND 10
.not between
: similar tobetween
except theBETWEEN
is replaced withNOT BETWEEN
in the generated condition.in
: operand 1 should be a column or DB expression, and operand 2 be an array representing the range of the values that the column or DB expression should be in. For example,['in', 'id', [1, 2, 3]]
will generateid IN (1, 2, 3)
. The method will properly quote the column name and escape values in the range.not in
: similar to thein
operator except thatIN
is replaced withNOT IN
in the generated condition.like
: operand 1 should be a column or DB expression, and operand 2 be a string or an array representing the values that the column or DB expression should be like. For example,['like', 'name', 'tester']
will generatename LIKE '%tester%'
. When the value range is given as an array, multipleLIKE
predicates will be generated and concatenated usingAND
. For example,['like', 'name', ['test', 'sample']]
will generatename LIKE '%test%' AND name LIKE '%sample%'
. You may also provide an optional third operand to specify how to escape special characters in the values. The operand should be an array of mappings from the special characters to their escaped counterparts. If this operand is not provided, a default escape mapping will be used. You may usefalse
or an empty array to indicate the values are already escaped and no escape should be applied. Note that when using an escape mapping (or the third operand is not provided), the values will be automatically enclosed within a pair of percentage characters.or like
: similar to thelike
operator except thatOR
is used to concatenate theLIKE
predicates when operand 2 is an array.not like
: similar to thelike
operator except thatLIKE
is replaced withNOT LIKE
in the generated condition.or not like
: similar to thenot like
operator except thatOR
is used to concatenate theNOT LIKE
predicates.exists
: requires one operand which must be an instance of [[yii\db\Query]] representing the sub-query. It will build aEXISTS (sub-query)
expression.not exists
: similar to theexists
operator and builds aNOT EXISTS (sub-query)
expression.
If you are building parts of condition dynamically it's very convenient to use andWhere
and orWhere
:
$status = 10;
$search = 'yii';
$query->where(['status' => $status]);
if (!empty($search)) {
$query->andWhere(['like', 'title', $search]);
}
In case $search
isn't empty the following SQL will be generated:
WHERE (`status` = 10) AND (`title` LIKE '%yii%')
For ordering results orderBy
and addOrderBy
could be used:
$query->orderBy([
'id' => SORT_ASC,
'name' => SORT_DESC,
]);
Here we are ordering by id
ascending and then by name
descending.
### `GROUP BY` and `HAVING`
In order to add `GROUP BY` to generated SQL you can use the following:
```php
$query->groupBy('id, status');
If you want to add another field after using groupBy
:
$query->addGroupBy(['created_at', 'updated_at']);
To add a HAVING
condition the corresponding having
method and its andHaving
and orHaving
can be used. Parameters
for these are similar to the ones for where
methods group:
$query->having(['status' => $status]);
To limit result to 10 rows limit
can be used:
$query->limit(10);
To skip 100 fist rows use:
$query->offset(100);
The JOIN
clauses are generated in the Query Builder by using the applicable join method:
innerJoin()
leftJoin()
rightJoin()
This left join selects data from two related tables in one query:
$query->select(['tbl_user.name AS author', 'tbl_post.title as title'])
->from('tbl_user')
->leftJoin('tbl_post', 'tbl_post.user_id = tbl_user.id');
In the code, the leftJoin()
method's first parameter
specifies the table to join to. The second parameter defines the join condition.
If your database application supports other join types, you can use those via the generic join
method:
$query->join('FULL OUTER JOIN', 'tbl_post', 'tbl_post.user_id = tbl_user.id');
The first argument is the join type to perform. The second is the table to join to, and the third is the condition.
Like FROM
, you may also join with sub-queries. To do so, specify the sub-query as an array
which must contain one element. The array value must be a Query
object representing the sub-query,
while the array key is the alias for the sub-query. For example,
$query->leftJoin(['u' => $subQuery], 'u.id=author_id');
UNION
in SQL adds results of one query to results of another query. Columns returned by both queries should match.
In Yii in order to build it you can first form two query objects and then use union
method:
$query = new Query;
$query->select("id, 'post' as type, name")->from('tbl_post')->limit(10);
$anotherQuery = new Query;
$anotherQuery->select('id, 'user' as type, name')->from('tbl_user')->limit(10);
$query->union($anotherQuery);
When working with large amount of data, methods such as [[yii\db\Query::all()]] are not suitable because they require loading all data into the memory. To keep the memory requirement low, Yii provides the so-called batch query support. A batch query makes uses of data cursor and fetches data in batches.
Batch query can be used like the following:
use yii\db\Query;
$query = (new Query)
->from('tbl_user')
->orderBy('id');
foreach ($query->batch() as $users) {
// $users is an array of 100 or fewer rows from the user table
}
// or if you want to iterate the row one by one
foreach ($query->each() as $user) {
// $user represents one row of data from the user table
}
The method [[yii\db\Query::batch()]] and [[yii\db\Query::each()]] return an [[yii\db\BatchQueryResult]] object
which implements the Iterator
interface and thus can be used in the foreach
construct.
During the first iteration, a SQL query is made to the database. Data are since then fetched in batches
in the iterations. By default, the batch size is 100, meaning 100 rows of data are being fetched in each batch.
You can change the batch size by passing the first parameter to the batch()
or each()
method.
Compared to the [[yii\db\Query::all()]], the batch query only loads 100 rows of data at a time into the memory. If you process the data and then discard it right away, the batch query can help keep the memory usage under a limit.
If you specify the query result to be indexed by some column via [[yii\db\Query::indexBy()]], the batch query will still keep the proper index. For example,
use yii\db\Query;
$query = (new Query)
->from('tbl_user')
->indexBy('username');
foreach ($query->batch() as $users) {
// $users is indexed by the "username" column
}
foreach ($query->each() as $username => $user) {
}