Количество строк в Doctrine QueryBuilder - PullRequest
181 голосов
/ 09 февраля 2012

Я использую QueryBuilder от Doctrine для создания запроса и хочу получить общее количество результатов запроса.

$repository = $em->getRepository('FooBundle:Foo');

$qb = $repository->createQueryBuilder('n')
        ->where('n.bar = :bar')
        ->setParameter('bar', $bar);

$query = $qb->getQuery();

//this doesn't work
$totalrows = $query->getResult()->count();

Я просто хочу подсчитать этот запрос, чтобы получить общее количество строк, но не возвращать фактические результаты. (После этого запроса количества я собираюсь изменить запрос с помощью maxResults для разбивки на страницы.)

Ответы [ 9 ]

443 голосов
/ 09 февраля 2012

Что-то вроде:

$qb = $entityManager->createQueryBuilder();
$qb->select('count(account.id)');
$qb->from('ZaysoCoreBundle:Account','account');

$count = $qb->getQuery()->getSingleScalarResult();

EDIT

Некоторые люди считают, что выражения лучше чем просто использование DQL. Один даже зашел так далеко, что отредактировал четырехлетний ответ. Я откатил его правку обратно. Пойди разберись.

47 голосов
/ 22 ноября 2014

Вот еще один способ отформатировать запрос:

return $repository->createQueryBuilder('u')
            ->select('count(u.id)')
            ->getQuery()
            ->getSingleScalarResult();
19 голосов
/ 11 ноября 2015

Лучше перенести всю логику работы с базой данных на репозитории.

Так в контроллере вы пишете

/* you can also inject "FooRepository $repository" using autowire */
$repository = $this->getDoctrine()->getRepository(Foo::class);
$count = $repository->count();

А в Repository/FooRepository.php

public function count()
{
    $qb = $repository->createQueryBuilder('t');
    return $qb
        ->select('count(t.id)')
        ->getQuery()
        ->getSingleScalarResult();
}

Лучше переместить $qb = ... в отдельную строку, если вы хотите создать сложные выражения, такие как

public function count()
{
    $qb = $repository->createQueryBuilder('t');
    return $qb
        ->select('count(t.id)')
        ->where($qb->expr()->isNotNull('t.fieldName'))
        ->andWhere($qb->expr()->orX(
            $qb->expr()->in('t.fieldName2', 0),
            $qb->expr()->isNull('t.fieldName2')
        ))
        ->getQuery()
        ->getSingleScalarResult();
}

Также подумайте о кэшировании результата запроса - http://symfony.com/doc/current/reference/configuration/doctrine.html#caching-drivers

public function count()
{
    $qb = $repository->createQueryBuilder('t');
    return $qb
        ->select('count(t.id)')
        ->getQuery()
        ->useQueryCache(true)
        ->useResultCache(true, 3600)
        ->getSingleScalarResult();
}

В некоторых простых случаях использование EXTRA_LAZY отношений сущностей хорошо
http://doctrine -orm.readthedocs.org / проекты / Доктрина-ОРМ / о / последние / учебники / экстра-ленивый-associations.html

16 голосов
/ 22 августа 2014

Если вам нужно посчитать более сложный запрос, с groupBy, having и т. Д ... Вы можете позаимствовать у Doctrine\ORM\Tools\Pagination\Paginator:

$paginator = new \Doctrine\ORM\Tools\Pagination\Paginator($query);
$totalRows = count($paginator);
6 голосов
/ 29 июня 2015

Пример работы с группировкой, объединением и прочим.

Проблема:

 $qb = $em->createQueryBuilder()
     ->select('m.id', 'rm.id')
     ->from('Model', 'm')
     ->join('m.relatedModels', 'rm')
     ->groupBy('m.id');

Для того, чтобы это работало, возможное решение - использовать специальный гидратор, и эта странная вещь называется «НАИМЕНОВАНИЕ ПО ВЫБОРУ ПО ВЫБОРУ»:

class CountHydrator extends AbstractHydrator
{
    const NAME = 'count_hydrator';
    const FIELD = 'count';

    /**
     * {@inheritDoc}
     */
    protected function hydrateAllData()
    {
        return (int)$this->_stmt->fetchColumn(0);
    }
}
class CountSqlWalker extends SqlWalker
{
    /**
     * {@inheritDoc}
     */
    public function walkSelectStatement(AST\SelectStatement $AST)
    {
        return sprintf("SELECT COUNT(*) AS %s FROM (%s) AS t", CountHydrator::FIELD, parent::walkSelectStatement($AST));
    }
}

$doctrineConfig->addCustomHydrationMode(CountHydrator::NAME, CountHydrator::class);
// $qb from example above
$countQuery = clone $qb->getQuery();
// Doctrine bug ? Doesn't make a deep copy... (as of "doctrine/orm": "2.4.6")
$countQuery->setParameters($this->getQuery()->getParameters());
// set custom 'hint' stuff
$countQuery->setHint(Query::HINT_CUSTOM_OUTPUT_WALKER, CountSqlWalker::class);

$count = $countQuery->getResult(CountHydrator::NAME);
5 голосов
/ 08 мая 2018

С Doctrine 2.6 можно использовать метод count() непосредственно из EntityRepository. Подробнее см. Ссылку.

https://github.com/doctrine/doctrine2/blob/77e3e5c96c1beec7b28443c5b59145eeadbc0baf/lib/Doctrine/ORM/EntityRepository.php#L161

4 голосов
/ 26 июля 2016

Для подсчета элементов после некоторого количества элементов (смещение) в этом случае нельзя применить $ qb-> setFirstResults (), поскольку оно работает не как условие запроса, а как смещение результата запроса для диапазона выбранные элементы (то есть setFirstResult не может использоваться вместе с COUNT вообще). Поэтому для подсчета оставшихся предметов я просто сделал следующее:

   //in repository class:
   $count = $qb->select('count(p.id)')
      ->from('Products', 'p')
      ->getQuery()
      ->getSingleScalarResult();

    return $count;

    //in controller class:
    $count = $this->em->getRepository('RepositoryBundle')->...

    return $count-$offset;

Кто-нибудь знает более чистый способ сделать это?

4 голосов
/ 02 декабря 2015

Для людей, которые используют только Doctrine DBAL, а не Doctrine ORM, они не смогут получить доступ к методу getQuery(), потому что он не существует. Им нужно сделать что-то вроде следующего.

$qb = new QueryBuilder($conn);
$count = $qb->select("count(id)")->from($tableName)->execute()->fetchColumn(0);
0 голосов
/ 14 февраля 2017

Добавление следующего метода в ваш репозиторий должно позволить вам вызвать $repo->getCourseCount() из вашего контроллера.

/**
 * @return array
 */
public function getCourseCount()
{
    $qb = $this->getEntityManager()->createQueryBuilder();

    $qb
        ->select('count(course.id)')
        ->from('CRMPicco\Component\Course\Model\Course', 'course')
    ;

    $query = $qb->getQuery();

    return $query->getSingleScalarResult();
}
...