首页 > 解决方案 > 在 cakephp 3 中添加包含模型的多个计数字段以进行查询

问题描述

我想在我的模型中包含另一个模型包含的完整模型条目集,而不是它们的数量。我可以通过将“大小”字段添加到结果集中来做到这一点,但我想在查询中这样做,因为我想对结果进行分页和排序。我正在使用此代码,但由于某种原因,如果第一个计数不为零,则两个计数字段具有相同的值,即第二个计数字段的值。

$query = $this->Users->find('all')
            ->contain(['Likes','Favs']);

 $query
      ->select(['like_count' => $query->func()->count('Likes.id')])
      ->leftJoinWith('Likes')
      ->group(['Users.id'])
      ->autoFields(true);

 $query
     ->select(['fav_count' => $query->func()->count('Favs.id')])
     ->leftJoinWith('Favs')
     ->group(['Users.id'])
     ->autoFields(true);

$this->paginate['sortWhitelist'] = [
            'name',
            'email',
            'last_login',
            'fav_count',
            'like_count',
        ];

我想知道为什么会发生这种情况以及是否有任何其他方法可以做我尝试的事情,那就是拥有('name'、email、'last_login'、带有用户 ID 的 Likes 中的条目数量、条目数量在带有用户 ID 的收藏夹中)。我曾尝试使用 join() 进行左连接,但无法获得我想要的结果。

标签: mysqljoincakephpcountcakephp-3.x

解决方案


如果您有多个联接,则需要使用 计数DISTINCT,例如:

COUNT(DISTINCT Likes.id)

这是因为您的结果将包含Likes * Favs行数,因为对于每个加入的点赞,所有收藏夹都将被加入,即对于2喜欢和10收藏夹,您最终会得到20总计行。常规COUNT()将包括这些行中的每一行。

另请注意,您不需要重复所有分组等内容,并且可以使用可调用的 forselect()来避免破坏构建器。

$query = $this->Users
    ->find('all')
    ->select(function (\Cake\ORM\Query $query) {
        return [
            'like_count' => $query->func()->count(
                $query->func()->distinct(['Likes.id' => 'identifier'])
            ),
            'fav_count' => $query->func()->count(
                $query->func()->distinct(['Favs.id' => 'identifier'])
            ),
        ];
    })
    ->autoFields(true)
    ->contain(['Likes', 'Favs'])
    ->leftJoinWith('Likes')
    ->leftJoinWith('Favs')
    ->group(['Users.id']);

使用函数生成器生成DISTINCT是一种解决方法,因为目前还没有 API 可以专门生成关键字。结果将类似于DISTINCT(Likes.id),但它会正常工作,括号将被解释为关键字之后表达式的一部分,而不是函数调用的一部分。


推荐阅读