Laravel 提供了 Eager Loading 使用 with()
方法来缓解 N+1 的问题,但是在实际使用中还是存在一些问题的,with()
会直接查询出表中所有的字段,而我们可能仅仅需要其中指定的某几个字段。
假如我们现在有两张表:user
和 posts
,每个 User 可以拥有多个 Posts,而每一篇 Post 只能属于一个 User。下面分别是 User Model 和 Post Model 中定义的关系:
// User.php public function post(){ return $this->hasMany('post'); } // Post.php public function user(){ return $this->belongsTo('user'); }
通过 with()
方法,我们可以这样获取所有的 Posts 并同时查出对应的 User 信息,可以使用这样的方法:
public function getAllPosts() { return Post::with('user')->get(); }
这实际上会执行下面两句 SQL 语句:
select * from `posts` select * from `users` where `users`.`id` in (<1>, <2>)
但是我们可能并不需要 User 表中所有的字段,例如我们只需要 id
和 username
两个字段:
select * from `posts` select id,username from `users` where `users`.`id` in (<1>, <2>)
我们可以通过下面两种方法来实现。
方法一:在 with() 中指定
Post::with(array('user'=>function($query){ $query->select('id','username'); }))->get();
方法二:修改 Model 文件
修改 Post.php 文件中 user() 方法:
public function user() { return $this->belongsTo('User')->select(array('id', 'username')); }
该篇属于专题:《Laravel小技巧》
- 下一篇:《Laravel 实现获取上一条/下一条记录的功能》
- 上一篇:《Laravel 实现多字段登录》
php 2016/03/19 16:15
一直用第一种
Silov 2016/07/20 14:51
这种方法有限制是哪个版本的Laravel才能用么?为什么我的不行。。。
Specs 2016/07/22 21:21
@ 我测试过 5.1 ,你的什么版本?
Silov 2016/07/25 15:08
@ 今天又试了下可以用了。。。上次不知道哪儿抽了。。。同5.1,谢谢~
旒侃 2017/07/18 16:58
@ 你知道是什么原因导致你不能用吗?我也遇到了同样的情况。
路人甲 2018/12/11 17:01
第一种可以简化成:Post::with(array('user:id, username'))->get();我是5.5,其余版本不知道。
Specs 2018/12/11 19:54
@ 大家用到的可以测试下