Eloquent ORM 实例教程 —— 查询作用域和模型事件


guide-to-eloquent-orm

1、查询作用域

Eloquent还支持将一些常用的查询封装到模型方法中,方便调用,我们将其称之为“查询作用域”,实现查询作用域很简单,只需要在模型方法前加上scope前缀即可,比如我们经常需要获取浏览数最高的文章,就可以使用该机制实现——在Post中定义一个scopePopular方法:

  1. public function scopePopular($query)
  2. {
  3. return $query->where('views','>=',100);
  4. }

对应的,我们在控制器中定义测试代码如下:

  1. $posts = Post::popular()->orderBy('views','desc')->get();
  2. foreach ($posts as $post) {
  3. echo '&lt;'.$post->title.'&gt; '.$post->views.'views<br>';
  4. }

在浏览器中访问http://laravel.app:8000/test,输出如下:

  1. <test 3> 800views
  2. <test 2> 500views
  3. <test 1 title> 100views

此外,查询作用域还支持动态传入参数,为了测试该方法我们为posts新增一个status字段:

为posts表添加status字段

同时在模型类中新增一个scopeStatus方法:

  1. public function scopeStatus($query,$status=1)
  2. {
  3. return $query->where('status',$status);
  4. }

接下来测试下该方法:

  1. $posts = Post::popular()->status(1)->orderBy('views','desc')->get();
  2. foreach ($posts as $post) {
  3. echo '&lt;'.$post->title.'&gt; '.$post->views.'views<br>';
  4. }

对应输出如下:

  1. <test 3> 800views
  2. <test 2> 500views

2、模型事件

Eloquent也支持模型事件——当模型被创建、更新或删除的时候触发相应事件,Eloquent目前支持八种事件类型:creatingcreatedupdatingupdatedsavingsaveddeletingdeleted

deletingdeleted很好理解,在删除模型时触发,deleting在删除操作前执行,deleted在删除完成后执行。

当创建模型时,依次执行savingcreatingcreatedsaved,同理在更新模型时依次执行savingupdatingupdatedsaved。无论是使用批量赋值(create/update)还是直接调用save方法,都会触发对应事件(前提是注册了相应的模型事件)。

你可以在任何你喜欢的地方注册模型事件,这里我们选择在服务提供者AppServiceProviderboot方法中注册:

  1. Post::saving(function($post){
  2. echo 'saving event is fired<br>';
  3. });
  4.  
  5. Post::creating(function($post){
  6. echo 'creating event is fired<br>';
  7. });
  8.  
  9. Post::created(function($post){
  10. echo 'created event is fired<br>';
  11. });
  12.  
  13. Post::saved(function($post){
  14. echo 'saved event is fired<br>';
  15. });

然后在控制器中编写测试代码如下:

  1. $data = array(
  2. 'title'=>'test model event',
  3. 'content'=>'test content',
  4. 'cat_id'=>1,
  5. );
  6. $post = Post::create($data);
  7. if(!$post->exists){
  8. echo '添加文章失败!';exit();
  9. }
  10. echo '&lt;'.$post->title.'&gt;保存成功!';

接下来在浏览中访问http://laravel.app:8000/test,页面输入如下:

  1. saving event is fired
  2. creating event is fired
  3. created event is fired
  4. saved event is fired
  5. <test model event>保存成功!

需要注意的是如果saving/creating/updating/deleting事件返回false,则相应的创建/更新/删除操作会退出,不再往下执行,比如我们修改上述creating事件代码如下:

  1. Post::creating(function($post){
  2. echo 'creating event is fired<br>';
  3. if($post->cat_id==1)
  4. return false;
  5. });

也就是当文章分类id等于1的时候,不再往下执行,在浏览器中再次访问http://laravel.app:8000/test,页面输出如下:

  1. saving event is fired
  2. creating event is fired
  3. 添加文章失败!

有了模型事件之后,我们就很方便地在模型创建、更新或删除的不同生命周期阶段添加相应的业务逻辑。