欢迎各位兄弟 发布技术文章

这里的技术是共享的

You are here

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

shiping1 的头像

1、查询作用域

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

public function scopePopular($query)
{
    return $query->where('views','>=',100);
}

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

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

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

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

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

为posts表添加status字段

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

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

接下来测试下该方法:

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

对应输出如下:

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

2、模型事件

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

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

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

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

Post::saving(function($post){
    echo 'saving event is fired<br>';
});

Post::creating(function($post){
    echo 'creating event is fired<br>';
});

Post::created(function($post){
    echo 'created event is fired<br>';
});

Post::saved(function($post){
    echo 'saved event is fired<br>';
});

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

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

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

saving event is fired
creating event is fired
created event is fired
saved event is fired
<test model event>保存成功!

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

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

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

saving event is fired
creating event is fired
添加文章失败!

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

  • 福哥在码代码
    福哥在码代码

    学院君,我把关于点坐标存入数据库,通过软件查询数据库,数据确实已经存进去,但是我想删除该点时候却删除不了,原因是该点不在数据库,但是如果刷新网页后去删除就可以删除成功。(用ajax post数据到后台的)

  • 龙腾网络工作室
    1. 龙腾网络工作室
      好几天没新文章了,我建议翻译写扩展组件吧!例如dingo/api,还有个比较重要的,就是laravel里怎么根据url让导航是选中状态,似乎是hieu-le/active       
    2. 学院君
      因为学院君去医院陪老婆生孩子去了 今天我回来了 [哈哈]        

    恭喜

  • 星空
    1. 龙腾网络工作室
      好几天没新文章了,我建议翻译写扩展组件吧!例如dingo/api,还有个比较重要的,就是laravel里怎么根据url让导航是选中状态,似乎是hieu-le/active       
    2. 学院君
      因为学院君去医院陪老婆生孩子去了 今天我回来了 [哈哈]        

    恭喜恭喜啊

  • 学院君
    1. 龙腾网络工作室
      好几天没新文章了,我建议翻译写扩展组件吧!例如dingo/api,还有个比较重要的,就是laravel里怎么根据url让导航是选中状态,似乎是hieu-le/active       

    因为学院君去医院陪老婆生孩子去了 今天我回来了 [哈哈]

  • 学院君
    1. 复写纸
      学院君。又来请教你了,请问composer.json里面已经用 "psr-4": {
      "App\\": "app/"了来规定自动加载app文件夹下面的东西了,为什么每次加入自己的类还需要dump_autoload呢。优化我可以理解,可是不加这个完全找不到文件啊       
    2. 还有1条评论
    3. 星空
      学院君,什么时候也出篇文章说说composer的相关知识和使用嘛~~       

    感觉这个网站已经基本可以满足日常使用需要了:http://www.phpcomposer.com/,我也就没必要重复造轮子了

  • 龙腾网络工作室

    好几天没新文章了,我建议翻译写扩展组件吧!例如dingo/api,还有个比较重要的,就是laravel里怎么根据url让导航是选中状态,似乎是hieu-le/active

  • 星空
    1. 复写纸
      学院君。又来请教你了,请问composer.json里面已经用 "psr-4": {
      "App\\": "app/"了来规定自动加载app文件夹下面的东西了,为什么每次加入自己的类还需要dump_autoload呢。优化我可以理解,可是不加这个完全找不到文件啊       
    2. 学院君
      你自己的类是放在哪里的       

    学院君,什么时候也出篇文章说说composer的相关知识和使用嘛~~

  • 学院君
    1. 徐自立
      Post::created(function($post){
      echo 'saved event is fired
      ';
      });
      这里是saved 吧       

    [汗] 确实 已修正

  • 徐自立

    Post::created(function($post){
    echo 'saved event is fired
    ';
    });
    这里是saved 吧

  • 学院君
    1. 复写纸
      学院君。又来请教你了,请问composer.json里面已经用 "psr-4": {
      "App\\": "app/"了来规定自动加载app文件夹下面的东西了,为什么每次加入自己的类还需要dump_autoload呢。优化我可以理解,可是不加这个完全找不到文件啊       

    你自己的类是放在哪里的


来自 http://laravelacademy.org/post/1073.html
普通分类: