我如何知道OneToMany关系已被更改?

时间:2013-06-03 17:10:35

标签: symfony doctrine-orm

我有以下实体:

class Person
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * 
     * @ORM\OneToMany(targetEntity="PersonEmail", mappedBy="person", cascade={"persist", "remove"})
     */
    private $emails;

    /**
     * @var datetime $updated
     *
     * @ORM\Column(type="datetime")
     */
    private $updated;
}

class PersonEmail
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @var string
     *
     * @ORM\Column(name="email", type="string", length=255)
     */
    private $email;

    /**
     * @ORM\ManyToOne(targetEntity="Person", inversedBy="emails")
     * @ORM\JoinColumn(name="person_id", referencedColumnName="id", onDelete="CASCADE")
     */
    private $person;
}

我使用Symfony2表单构建器来创建表单。表单提交后,我致电$form->bind($request)。我的问题是:在保存Person实体之前,我怎么知道电子邮件被更改(插入,更新或删除)?我需要它,因为我想更新Person :: $ updated字段。我可以在使用PreUpdate生命周期回调修改Person实体时执行此操作,但是如果仅更改电子邮件列表,如何更新此字段?预测最简单的答案:我想避免在没有任何改变的情况下更新此字段。

不可能在PersonEmail的生命周期回调中更新Person实体,因为此时已经处理了Person实体。

控制器代码(不确定是否有帮助):

/**
 * 
 * @Template()
 */
public function editAction($id, Request $request)
{
    $repository = $this->getDoctrine()->getRepository('AcmeBundle:Person');
    $person = $repository->find($id);
    if (!$person) {
        throw $this->createNotFoundException('No person found for id '.$id);
    }
    if ($request->isMethod('POST')) {
        $form->bind($request);
        if ($form->isValid()) {
            $em = $this->getDoctrine()->getEntityManager();
            $em->flush();
        }
    }
    return array('form' => $form->createView(), 'person' => $person);
}

1 个答案:

答案 0 :(得分:0)

这是我的解决方案。它并不像我希望的那样优雅,但无论如何至少它有效。因此,在我的解决方案中,我们首先检查电子邮件集合是否脏(意味着添加和/或删除了一些电子邮件)。如果它不脏,我们仍然需要检查是否对现有电子邮件进行了任何更改。我本来希望使用一些现有的API,但似乎在这种情况下没有它们。所以,我决定手动检查我的数据是否有变化。代码示例如下。

/**
 * 
 * @Template()
 */
public function editAction($id, Request $request)
{
    $repository = $this->getDoctrine()->getRepository('AcmeBundle:Person');
    $person = $repository->find($id);
    if (!$person) {
        throw $this->createNotFoundException('No person found for id '.$id);
    }
    if ($request->isMethod('POST')) {
        $form->bind($request);
        if ($form->isValid()) {
            $em = $this->getDoctrine()->getEntityManager();

            // isDirty() will be true if emails were added or deleted
            if ($person->getEmails()->isDirty()) {
                $person->setUpdated(new \DateTime(date('Y-m-d H:i:s')));
            } else {
                $uow = $em->getUnitOfWork();

                // as no one suggested a working solution, I just decided
                // to check entity by entity, field by field
                // if anything is changed, we'll mark $person as updated
                foreach($person->getEmails() as $item) {
                    $orig = $uow->getOriginalEntityData($item);
                    if ($item->getEmail() != $orig['email']) {
                        $person->setUpdated(new \DateTime(date('Y-m-d H:i:s')));
                        break;
                    }
                }
            }
            $em->flush();
        }
    }
    return array('form' => $form->createView(), 'person' => $person);
}

如果有人知道如何避免逐个实体检查,如果您分享这样的解决方案,我将不胜感激。