Symfony2/Doctrine:如何将OneToMany实体重新保存为级联新行

div*_*oZA 18 php doctrine clone symfony doctrine-orm

首先,这个问题类似于如何将实体重新保存为Doctrine 2中的另一行

不同之处在于我正在尝试将数据保存在具有OneToMany关系的实体中.我想将实体重新保存为父实体中的新行(在"一"侧),然后作为每个后续子项中的新行(在"多"侧).

我使用了一个非常简单的教室示例,其中包含许多学生,以保持简单.

所以我可能有Classroom = 1的ClassroomA,它有5个学生(ids 1到5).我想知道如何在Doctrine2中使用该实体并将其重新保存到数据库中(在潜在的数据更改之后)全部使用新ID并且原始行在持久化/刷新期间不受影响.

让我们首先定义我们的Doctrine实体.

课堂实体:

namespace Acme\TestBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;

/**
 * @ORM\Entity
 * @ORM\Table(name="classroom")
 */
class Classroom
{
    /**
     * @ORM\Id
     * @ORM\Column(type="integer")
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

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

   /**
     * @ORM\OneToMany(targetEntity="Pupil", mappedBy="classroom")
     */
    protected $pupils;

    public function __construct()
    {
        $this->pupils = new ArrayCollection();
    }       
    // ========== GENERATED GETTER/SETTER FUNCTIONS BELOW ============

}
Run Code Online (Sandbox Code Playgroud)

学生实体:

namespace Acme\TestBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;

/**
 * @ORM\Entity
 * @ORM\Table(name="pupil")
 */
class Pupil
{
    /**
     * @ORM\Id
     * @ORM\Column(type="integer")
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

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

    /**
     * @ORM\ManyToOne(targetEntity="Classroom", inversedBy="pupils")
     * @ORM\JoinColumn(name="classroom_id", referencedColumnName="id")
     */
    protected $classroom;   

    // ========== GENERATED FUNCTIONS BELOW ============
}
Run Code Online (Sandbox Code Playgroud)

我们的通用Action功能:

public function someAction(Request $request, $id)
{
    $em = $this->getDoctrine()->getEntityManager();

    $classroom = $em->find('AcmeTestBundle:Classroom', $id);

    $form = $this->createForm(new ClassroomType(), $classroom);

    if ('POST' === $request->getMethod()) {
        $form->bindRequest($request);

        if ($form->isValid()) {
            // Normally you would do the following:
            $em->persist($classroom);
            $em->flush();

            // But how do I create a new row with a new ID 
            // Including new rows for the Many side of the relationship

            // ... other code goes here.
        }
    }

    return $this->render('AcmeTestBundle:Default:index.html.twig');
}
Run Code Online (Sandbox Code Playgroud)

我尝试过使用克隆,但只保存了父关系(在我们的示例中是Classroom),并使用新ID,而子数据(Pupils)则根据原始ID进行了更新.

在此先感谢任何帮助.

Phi*_*hil 30

事情clone是......

克隆对象时,PHP 5将执行所有对象属性的浅表副本.任何引用其他变量的属性都将保留引用.

如果您使用的是Doctrine> = 2.0.2,则可以实现自己的自定义__clone()方法:

public function __clone() {
    // Get current collection
    $pupils = $this->getPupils();

    $this->pupils = new ArrayCollection();
    foreach ($pupils as $pupil) {
        $clonePupil = clone $pupil;
        $this->pupils->add($clonePupil);
        $clonePupil->setClassroom($this);
    }
}
Run Code Online (Sandbox Code Playgroud)

注意:在Doctrine 2.0.2之前,您无法__clone()在实体中实现方法,因为生成的代理类实现了自己的方法__clone(),而不检查或调用parent::__clone().因此,您必须为clonePupils()(in Classroom)创建一个单独的方法,并在克隆实体后调用它.无论哪种方式,您都可以在您__clone()clonePupils()方法中使用相同的代码.

克隆父类时,此函数也将创建一个充满子对象克隆的新集合.

$cloneClassroom = clone $classroom;
$cloneClassroom->clonePupils();

$em->persist($cloneClassroom);
$em->flush();
Run Code Online (Sandbox Code Playgroud)

您可能希望级联持久化您的$pupils集合以使持久化更容易,例如

/**
 * @ORM\OneToMany(targetEntity="Pupil", mappedBy="classroom", cascade={"persist"})
 */
protected $pupils;
Run Code Online (Sandbox Code Playgroud)

  • 这样做了,谢谢.我希望避免在函数中循环,但这是最简单的解决方案.我看到Doctrine的EntityManager类有一个"复制"功能,描述为"创建给定实体的副本.可以创建浅拷贝或深拷贝".唯一的问题是函数有一行抛出异常"未实现". (4认同)
  • 在当前的Doctrine2中,您可以实现自己的`__clone`,它将被调用. (2认同)