Symfony - 更新多个记录

时间:2012-01-06 20:26:28

标签: optimization doctrine symfony

使用doctrine,symfony2更新数据库中的多条记录的最佳方法是什么?

我收到一些记录ID,我必须更新 我想为每个记录分配从接收到的数组到列show_order的索引。 所以,如果我收到数组$ array = array(22,1,5,10),那么我想做

 $i = 0;
 foreach($array as $a) {
    $record = $this->getDoctrine->getRepository('AcmeBundle:SomeEntity')->findOneById($a);
    if ($record != null) $record->setOrder($i++);
 }
 $this->getDoctrine()->getEntityManager()->flush();

但这很可怕,因为对于每条记录我都做一个SELECT,所以查询数量是O(n)。

如何做得更好?

3 个答案:

答案 0 :(得分:9)

像...一样的东西。

foreach ($repo->findById($ids) as $obj) {
    $obj->setOrder(array_search($obj->getId(), $ids));
}

$em->flush();

答案 1 :(得分:7)

作为第一种选择,您应该考虑Batch Processing。如果由于某种原因这对你不可行,第二种选择是使用普通的SQL,可能是DBAL

答案 2 :(得分:1)

所以这仍然是0(n),但它是1n而不是2n。为了避免不必要的选择,我使用custom repository class和doctrine查询构建器解决了这个问题:

namespace BRS\PageBundle\Repository;

use Doctrine\ORM\EntityRepository;

class ContentRepository extends EntityRepository
{
    public function reorder($content)
    {    
        $em = $this->getEntityManager();

        $count = 0;

        foreach($content as $i => $content_id){

            $q = $em->createQuery('update BRSPageBundle:Content c set c.display_order = ?1 where c.id = ?2')
                    ->setParameter(1, $i)
                    ->setParameter(2, $content_id);

            $count += $q->execute();
        }

        return $count;
    }
}

然后说你有这样的内容id数组:

$content = array(23,12,8,4);

然后您可以非常简单地从控制器更新订单:

$count = $this->getRepository('BRSPageBundle:Content')->reorder($content);