我正在看Scracch系列中的Laracasts:Laravel 5.4,并且遇到了查询范围的概念。
在视频中,我们设置了这样一个类:
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Task extends Model
{
public function scopeIncomplete($query)
{
return $query->where('completed', 0);
}
}
我的第一个问题:
为什么这样的方法如下所示: public function scopeIncomplete($query)
这样调用App\Task::incomplete()
并使用::
运算符?
这不是静态调用的非静态方法吗?
我的第二个问题:
$query
来自哪里?
据我所知,这是“现有”查询,但它的调用方式如下:App\Task::incomplete()->where('id', '>', 1)->get();
所以我不确定这个变量的来源。
以后可能会对此进行更深入的解释,但我无法理解其工作原理。
答案 0 :(得分:3)
这是非常广泛的,因为它涵盖了Eloquent的两个大型主题。
第一个是指Eloquent使用Facade模式将静态调用传递给非静态方法。
Facade模式利用IoC容器将方法调用传递给类的绑定或新实例。
Facade模式使用重载和魔术方法__callStatic。
第二个问题涉及Eloquent的重载,它将未注册的方法调用传递给查询构建器对象(特别是\ Illuminate \ Database \ Eloquent \ Builder)。这允许在Eloquent模型本身上调用where()和各种其他Query Builder方法。在Model类中,您可以看到:
/**
* Handle dynamic method calls into the model.
*
* @param string $method
* @param array $parameters
* @return mixed
*/
public function __call($method, $parameters)
{
if (in_array($method, ['increment', 'decrement'])) {
return $this->$method(...$parameters);
}
return $this->newQuery()->$method(...$parameters);
}
由于未在Model中定义where()或incomplete(),因此它们将被传递给Builder类,该类由Model :: newQuery()返回。
在Builder __call方法中,你有:
if (method_exists($this->model, $scope = 'scope'.ucfirst($method))) {
return $this->callScope([$this->model, $scope], $parameters);
}
所以这是检查模型中定义的范围并传递包含此Builder实例的$参数。
总而言之,对Model :: incomplete()的调用将会出现:
模型__call() - &gt; Builder __call() - &gt;模型scopeIncomplete(Builder $ builder)