更新:公关已被合并到4.2,所以现在可以在方法( )中使用点嵌套符号,has
->has('relation1.relation2)
->whereHas('relation1.relation2, ..
您的问题仍然有些不清楚,或者您误以为Hass()方法用于过滤模型(在这种情况下为用户),并且仅获得具有适合搜索条件的相关模型的那些。
看来你想从给定的用户的上下文中找到Packages,所以不需要使用whereHas方法。
无论如何取决于关系(1-1,1-m,mm),这可以很容易或相当困难,而不是很有效率。正如我所说,加载嵌套关系意味着对于每个级别的嵌套来说都是另一个数据库查询,所以在这种情况下,最终得到5个查询。
不管关系如何,你可以这样反转这个链,因为这样会更容易:
编辑:这不会像atH那样工作atHas()不处理点嵌套关系!
// given $user and $search:
$packages = Package::where('alias','like',"%$search%")
->whereHas('product.membership.club.user', function ($q) use ($user) {
$q->whereId($user->id);
})->get();
你可以看到这是更可读的,仍然运行5个查询。也可以这样获得$包,这是您想要的模型的一个集合。
从使用者的角度来看,你会得到这样的东西(取决于关系再次):
$user
|-club
| |-membership
| | |-product
| | | |-packages
| | |-anotherProduct
| | | |-packages
| | |-yetAnotherProduct
| | |-packages
| |-anotherMembership
.....
你得到点,不是吗?
您可以从集合中获取包,但这将是麻烦的。其他方式更容易。
所以你的问题的答案只是加入表:
// Let's assume the relations are the easiest to handle: 1-many
$packages = Package::where('alias','like',"%$search%")
->join('products','packages.product_id','=','products.id')
->join('memberships','products.membership_id','=','memberships.id')
->join('clubs','memberships.club_id','=','clubs.id')
->where('clubs.user_id','=',$user->id)
->get(['packages.*']); // dont select anything but packages table
当然,你可以用一个很好的方法来包装它,所以你不必每次执行这样的搜索时都写这个。这个查询的性能肯定比以上显示的5个查询要好得多。显然这种方式你只加载包,没有其他相关的模型。