php - Eloquent 嵌套 whereHas

标签 php laravel laravel-4 eloquent

目前我的模型集合中有这个 whereHas:

$query = self::whereHas('club', function($q) use ($search)
{
    $q->whereHas('owner', function($q) use ($search)
    {
        $q->where('name', 'LIKE', '%'. $search .'%');
    });

});

我的印象是上面的代码可能是这样的:

$query = self::whereHas('club.owner', function($q) use ($search)
{
    $q->where('name', 'LIKE', '%'. $search .'%');    
});

我知道这已经很强大了,但即便如此,如果我有一个 5 层深的嵌套关系,事情会变得很糟糕。

更新:

如评论中所述,我最终没有把我的问题说清楚,我深表歉意。

我将尝试使用一个简单的示例,考虑 $owner->club->membership->product->package,现在我想从所有者那里搜索某个包,这将是一些东西像这样:

$query = self::whereHas('club', function($q) use ($search)
{
    $q->whereHas('membership', function($q) use ($search)
    {
        $q->whereHas('product', function($q) use ($search)
        {
            $q->whereHas('package', function($q) use ($search)
            {
                $q->where('alias', 'LIKE', '%'. $search .'%');
            });//package
        });//product
    });//membership
});//club

这是正确的吗?有捷径吗?

最佳答案

更新:PR 刚刚合并到 4.2,所以现在可以在 has 方法中使用 dot nested 表示法(->has ('relation1.relation2) ->whereHas('relation1.relation2, .. )

您的问题仍然有点不清楚,或者您误解了 whereHas() 方法,因为它用于过滤模型(在本例中为用户)并仅获取那些具有符合搜索条件的相关模型的模型。

您似乎想从给定的User 上下文中查找Packages,因此无需使用whereHas 方法。

无论如何,根据关系 (1-1,1-m,m-m),这可能很容易,也可能非常困难,而且效率不高。正如我所说,加载嵌套关系意味着对于每个嵌套级别都会有另一个数据库查询,因此在这种情况下,您最终会得到 5 个查询。

无论关系如何,您都可以像这样反转此链,因为这样会更容易:


编辑:这在 atm 上不起作用,因为 whereHas() 不处理点嵌套关系!

// 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 个查询。 同样,通过这种方式您可以获得 $packages,它是您想要的模型的单个集合。

虽然从用户的上下文中你会得到这样的东西(再次取决于关系):

$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 个查询好得多。显然,这种方式只加载包,没有其他相关模型。

关于php - Eloquent 嵌套 whereHas,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23153678/

相关文章:

php - 在 Laravel 4 中将数据传递给闭包

javascript - Laravel PHP Jquery - 文本框更新标签

php - 盐和密码

php - 保存belongsTo关系

php - 从 Laravel 4 中的数据库获取平均值

laravel - Laravel 4 原生 session 驱动程序如何工作?

php - 用数字更新列

PHP/Jquery -> 插件 : DataTables, 如何在数组中使用格式化程序?

mysql - is null 在 laravel 查询的末尾连接

php - 如何在 Laravel 而不是 Facade 上创建辅助方法