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

标签 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 实体在那一刻已经被处理。

Controller 代码(不确定是否有帮助):

/**
 * 
 * @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);
}

最佳答案

这是我的解决方案。它并不像我希望的那样优雅,但无论如何至少它是有效的。因此,在我的解决方案中,我们首先检查电子邮件集合是否脏(意味着添加和/或删除了一些电子邮件)。如果它不是脏的,我们仍然需要检查是否对现有电子邮件进行了任何更改。我希望为此使用一些现有的 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);
}

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

关于symfony - 我如何知道 OneToMany 关系已更改?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16902139/

相关文章:

php - symfony 表单选择字段 : set choices option default data from external datasource

php - Symfony2/Doctrine2 : How to override DebugStack class used in doctrine DataCollector?

php - 从 symfony 2 中的 Doctrine 2 实体中提取约束

php - symfony中的请求计数sql返回 bool 值而不是整数

mysql - 从表中获取列值作为 sql 查询结果中的键

php - Ratchet 服务器用户警告导致 Symfony ContextErrorException

symfony - Symfony 2:从存储库创建服务

php - Doctrine 2.1 和表 utf-8 编码

symfony - 学说查询构建器限制和偏移

php - Symfony2 AJAX 登录