使用查询器

到目前为止,我们还没有真正的使用查询器,之前我们在DB facade上面调用了一些简单方法,让我们构建一些查询.

查询器将方法链接到一起从而构建查询,在链的末尾你可以使用一些方法(可能是get())来触发查询.

来让我们看一个简单的例子

$usersOfType = DB::table('users')
        ->where('type', $type)
        ->get();

这里,我们创建了查询,查询users表格,查询条件是type,然后我们执行查询获取结果

让我们看下可用的链式方法有哪些,这些方法我称之为约束方法,修改方法,条件方法和结束/返回方法

约束方法

这些方法进行约束查询,并返回较小的数据子集

select()

选择查询的列

$emails = DB::table('contacts')
    ->select('email', 'email2 as second_email')
    ->get();
// Or
$emails = DB::table('contacts')
    ->select('email')
    ->addSelect('email2 as second_email')
    ->get();

where()

你可以使用WHERE限制返回内容的范围,默认情况下,where()接收3个参数,列,比较运算符,值

然而,如果第二个运算符是=,那么可以省略这个参数

$vipContacts = DB::table('contacts')->where('vip',true)->get();

如果要组合where()语句,你可以再另外一个后面追加或者传递数组

orWhere()

创建一个简单的OR WHERE

若要创建具有多个条件的更复杂OR WHERE语句,请给orWhere传递闭包

多个where()和orWhere()混合调用

如果将orWhere()调用与多个where()调用结合使用,则需要非常小心,以确保查询按您认为的方式进行。这并不是因为Laravel有任何错误,而是因为像下面这样的查询可能无法实现您期望的结果:

如果要编写SQL,上面的示例中明确指出“if this or(this and this)”,则需要将闭包传递到orWhere()中调用:

whereBetween(colName, [low, high])

允许查询返回介于两个值之间的行(包含两个值)

同样适用于whereNotBetween(),但是结果会反转

whereIn(colName, [1, 2, 3])

返回列表范围的行

同样适用于whereNotIn(),结果会反转

whereNull(colName) 和 whereNotNull(colName)

查询NULL或者Not NULL

whereRaw()

可以给where传递原生非转义字符串

小心SQL注入!

传递给whereraw()的任何SQL查询都不会被转义。小心使用此方法;这是应用程序中SQL注入攻击的主要机会。

whereExists()

允许你传递子查询,并至少返回一条记录,假设你想获取至少有一条留言的用户

distinct()

返回与其他记录比较后的记录,通常与select()搭配使用,因为如果你使用了主键,则不会有重复的记录

修改方法

这些方法更改了查询结果,不仅仅是限制了结果

orderBy(colName, direction)

查询结果排序,第二个参数可以是asc(默认值,升序)或desc(降序)

groupBy() 和 having() 或者 havingRaw()

将结果按列分组。或者,having()和havingraw()允许您根据组的属性筛选结果。例如,您可以只查找人口至少为30人的城市

skip()和take()

最常用于分页,它们允许您定义返回多少行,以及在开始返回之前跳过多少行,就像分页系统中的页码和页面大小一样。

latest(colName)和oldest(colName)

通过传递的列进行升序(oldest)或者降序(latest)排列,如果没传值则按照created_at排列

inRandomOrder()

随机排序结果

条件方法

Laravel 5.2以及更高版本有两个方法,可以通过传递值的布尔状态对内容进行条件判断(需要传递个闭包)

when()

第一个参数传"真"则会执行闭包内的查询,第一个参数传"假"那么什么都不做,注意第一个参数应该是布尔(true,false),一个可选值($status,从用户输入中提取,并默认为null)或返回一个值的闭包,重要的是要返回true或者false,例如

您还可以传递第三个参数,即另一个闭包,只有在第一个参数false时才会应用该闭包。

unless()

与when相反,如果第一个参数为false,则会运行第二个闭包

结束/返回方法

终止查询链并触发SQL查询,如果在查询链没有调用这些方法,那么你将得到一个查询器实例,如果在查询链上执行了这些方法,那么你将得到具体的记录.

get()

获取查询器的所有结果

first()和firstOrFail()

获取第一条结果,就像Like,但是它只返回一条

如果没有结果,first()将自动失败,而firstorfail()将引发异常。

如果将列名数组传递给任一方法,它将只返回这些列的数据,而不是所有列的数据。

find(id)和findOrFail(id)

与first()类似,传递一个与主键对应的ID进行查找,如果ID对应的不存在,find会失败,而findOrFail会异常

value()

从第一行提取单个字段,像first()但是你只想获取单个列

count()

返回所有匹配结果的计数

min()和max()

返回特定列的最小或最大值

sum()和avg()

返回特定列中所有值的总和或平均值

使用DB::raw在查询器中写原生查询

之前看到过一些方法用于原生语法-例如select有一个对应的selectRaw方法可以给其传递字符串用于查询,然后放到WHERE语法后

当然你也可以调用DB::raw()来实现相同的结果

连接

定义联合查询有时有点困难,但是Laravel的查询器做的比较好,如例子

join()创建了一个连接查询,你可以用链式写法组合多个join,或者使用leftJoin来创建左连接

最后,你也可以给join传递闭包创建更复杂的连接查询

组合

通过首先创建两个查询,然后使用union()或unionall()方法,可以将它们联合起来(将它们的结果组合成一个结果)

插入

插入非常简单,传递一个数组插入单条记录,传递数组的数组,插入多条记录,使用insertGetId()替代insert()可以自动生成主键

更新

更新也非常简单,创建更新查询只需要将数组传递给它

还可以使用increment()和decrement()方法快速增加和减少列。第一个参数是列名,第二个(可选)是要递增/递减的数字

删除

删除更简单,在查询结尾调用delete()即可

您还可以截断表,该表将删除每一行并重置自增ID。

JSON操作

如果你有一个JSON列,你可以使用箭头语法来更新或选择JSON结构

Last updated

Was this helpful?