手记

详解Laravel中Eloquent ORM的查询作用域

全局作用域:

    简单方便的为每个模型查询都加上约束条件。其中软删除功能就是利用该特性从数据库中获取未删除的模型。

代码演示:

    实现Illuminate\Database\Eloquent\Scope.php接口中apply方法。根据需求编写apply方法。

    

    在app下新建一个Scopes文件夹,然后创建ReadScope类。

    

    注意:如果需要用select语句准确获取对应字段的内容,应使用addSelect方法。防止替换现有select语句。

使用你所编写的全局作用域,那么就需要重写模型的boot方法并使用addGlobalScope方法:

    

当你使用Topic::all()时会生成如下SQL语句:

select * from `topic` where `read_num` > 200;

匿名全局作用域:不需要为了一个简单的作用域而编写一个单独的类

    

那怎么取消呢?

非匿名:Topic::withoutGlobalScope(ReadScope::class)->get();
匿名:Topic::withoutGlobalScope('read')->get();

取消所有用withoutGlobalScopes
Topic::withoutGlobalScopes()->get();
球
Topic::withoutGlobalScopes([
    ReadScope::class, WriteScope::class])->get();

本地作用域

    只需要在对应的Eloquent模型方法前添加scope前缀

    

如何使用呢?可以链式调用你所定义的所有本地作用域,但是调用的时候不需要加scope

    $topics = Topic::read()->top()->orderBy('topic_id')->get();
    闭包形式
    $topics = Topic::read()->orWhere(function (Builder $query){
        $query->top();
    })->get();
    闭包形式的高级用法:
    $topics = Topic::read()->orWhere->top()->get();

动态作用域:传参的形式

    

如何使用呢?

$topics = Topic::withOrder('created_at')->get();

重点介绍下$query的由来

介绍完了以上作用域,可能你应该还有点懵,很好奇$query是哪里跑出来的?文档中好像也只是提了怎么去用。这时该怎么办呢?文档又查不到。这时只能看源码了。来开始分析源码如下步骤:

    找到vendor/laravel/framework/src/Illuminate/Database/Eloquent/Builder.php这个文件中的__call()方法,你会发现有这么一段代码:

    

我们在上面已经定义了多个scope开头的方法,完全符合我们的预算。其会调用callScope方法,接下来我们就去会会这个callScope。

    我们用一丢丢的力气就找到了这个玩意:

    

源码解读:

array_unshift($parameters, $this);//将$this即当前的builder instance放在 $parameters 数组的第一个
$result = $scope(...array_values($parameters)) ?? $this;
回调。回调的是我们在上一步中调用callScope方法时的那个第一个参数所代表的方法,即我们定义在model中前
缀为scope的方法。array_values($parameters)是依次取出数组中的值返回一个数字数组。...$arr这种写法是将
$arr数组中的元素全部拆开依次放入对应的函数的参数列表。由于数组中第一个元素是当前的builder instance,
这样就将当前的builder instance传到了model中的前缀为scope方法的$query参数变量中,这样对$query的操作,
就是对当前的builder instance的操作了。因此后面的操作无非可以看成是SQL语句的一些order,where的操作。
return $result;   //返回的是一个eloquent builder instance.

慕友们你们看懂了嘛?点个赞支持下呗。

1人推荐
随时随地看视频
慕课网APP