Symfony2事件监听器preUpdate

时间:2015-11-10 18:12:24

标签: php symfony events

我已在 service.yml 中配置了一些事件监听器

registerproduct.listener:
        class: App\AppBundle\Listener\RegisterProductListener
        arguments: [ @service_container ]
        tags:
            - { name: doctrine.event_listener, event: prePersist }
            - { name: doctrine.event_listener, event: preUpdate }

我的方法 decrementProductQuantity()在我执行prePersist时效果很好,但在preUpdate时却没有。

在更新期间,产品数量会减少,但数据库中没有任何变化。

以下是我的事件代码:

public function prePersist(LifecycleEventArgs $args)
{
    $this->decrementProductQuantity($args);
}

public function preUpdate(LifecycleEventArgs $args)
{
    $this->decrementProductQuantity($args);
}

public function decrementProductQuantity(LifecycleEventArgs $args)
{
    /** @var RegisterProduct $registerProduct */
    $registerProduct = $args->getEntity();

    if (!($registerProduct instanceof RegisterProduct))
        return;

    if ($registerProduct->getStatus() == 'inscription') {
        $product = $registerProduct->getProduct();
        $product->setQuantityStock($product->getQuantityStock() - $registerProduct->getQuantity());
    }
}

以下是我的控制器中的更新代码:

/**
 * @param Register $register
 * @return \Symfony\Component\HttpFoundation\Response
 */
public function confirmAction(Request $request, Register $register)
{
    $em   = $this->getDoctrine()->getManager();

    foreach($register->getRegisterProducts() as $registerProduct){
        $registerProduct->setStatus('inscription');
        $em->persist($registerProduct);
        $em->flush();
    }       
    return $this->redirectWithMode($this->generateUrl('asc_activity_register_show', array('register' => $register->getId())));
}

我不知道为什么它在数据库中添加内容时有效,但不适用于更新。

感谢您的帮助。

3 个答案:

答案 0 :(得分:1)

在preUpdate侦听器上,您应该重新计算产品实体的变更集

public function decrementProductQuantity(LifecycleEventArgs $args)
{
    /** @var RegisterProduct $registerProduct */
    $registerProduct = $args->getEntity();

    if (!($registerProduct instanceof RegisterProduct))
        return;

    if ($registerProduct->getStatus() == 'inscription') {
        $product = $registerProduct->getProduct();
        $product->setQuantityStock($product->getQuantityStock() - $registerProduct->getQuantity());
        $em = $args->getEntityManager();
        $uow = $em->getUnitOfWork();
        $className = \Doctrine\Common\Util\ClassUtils::getClass($product);
        $classMetadata = $em->getClassMetadata($className);
        $uow->recomputeSingleEntityChangeSet($classMetadata, $product);
    }
}

这是因为在您的控制器(或其他地方)中,您的代码不会更新产品实体。在监听器执行doctrine之前,计算每个实体的更改集。因此,在您的 preUpdate 监听器中,对产品一无所知(因为它没有改变)。

答案 1 :(得分:0)

尝试更改preUpdate方法的签名:

use Doctrine\ORM\Event\PreUpdateEventArgs;

/.../

public function preUpdate(PreUpdateEventArgs $args)
{
    $this->decrementProductQuantity($args);
}

/.../

答案 2 :(得分:0)

因为我没有找到芒果,所以我最终选择直接在控制器中进行:

    /**
     * @param Register $register
     * @return \Symfony\Component\HttpFoundation\Response
     */
    public function confirmAction(Request $request, Register $register)
    {
        $mode = $this->getMode($request);
        $em   = $this->getDoctrine()->getManager();

        foreach ($register->getRegisterProducts() as $registerProduct) {
            if($registerProduct->getStatus() != 'inscription'){
                $product = $registerProduct->getProduct();
                $product->setQuantityStock($product->getQuantityStock() - $registerProduct->getQuantity());
            }

            $registerProduct->setStatus('inscription');
        }

        $em->flush();

        return $this->redirectWithMode($this->generateUrl('asc_activity_register_show', array('register' => $register->getId())), $mode);
    }

如果有人找到了anwser,请告诉我。