Symfony2 / Doctrine: как повторно сохранить объект с OneToMany в качестве каскадной новой строки

Во-первых, этот вопрос аналогичен тому, как переустановить объект как еще одну строку в Doctrine 2

Разница в том, что я пытаюсь сохранить данные в сущности, у которой есть отношения OneToMany. Я хотел бы повторно сохранить объект как новую строку в родительском объекте (на «одной» стороне), а затем в качестве новых строк в каждом последующем дочернем элементе (со стороны «много»).

Я использовал довольно простой пример класса, в котором было много учеников, чтобы это было просто.

Таким образом, у меня может быть ClassroomA с id = 1, и у него есть 5 учеников (от 1 до 5). Я хотел бы знать, как я мог, в Doctrine2, взять этот Entity и повторно сохранить его в базе данных (после потенциальных изменений данных), все с новыми идентификаторами повсюду, а исходные строки остаются нетронутыми во время persist / flush.

Давайте сначала определим наши сущности Учения.

Класс:

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 ============ } 

Ученик:

 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 ============ } 

И наша общая функция 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'); } 

Я попытался использовать клон, но только сохранил родительские отношения (класс в нашем примере) со свежим идентификатором, а данные детей (ученики) были обновлены по сравнению с исходными идентификаторами.

Заранее благодарим за любую помощь.

Related of "Symfony2 / Doctrine: как повторно сохранить объект с OneToMany в качестве каскадной новой строки"

Вещь с 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); } } 

ПРИМЕЧАНИЕ. До Doctrine 2.0.2 вы не можете реализовать метод __clone() в своей сущности, поскольку сгенерированный класс прокси реализует свой собственный __clone() который не проверяет или не вызывает parent::__clone() . Поэтому вам придется сделать отдельный метод для этого, например clonePupils()Classroom ), и вызвать это после того, как вы клонируете объект. В любом случае, вы можете использовать один и тот же код в своих методах __clone() или clonePupils() .

Когда вы клонируете свой родительский класс, эта функция создаст новую коллекцию, полную клонов дочерних объектов.

 $cloneClassroom = clone $classroom; $cloneClassroom->clonePupils(); $em->persist($cloneClassroom); $em->flush(); 

Вероятно, вы захотите, чтобы каскад сохранялся в коллекции ваших $pupils чтобы упростить работу, например

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

Я сделал это так, и все работает отлично.

Внутри клонированной сущности у нас есть волшебство __clone () . Там мы также не забываем о наших единодушных .

 /** * Clone element with values */ public function __clone(){ // we gonna clone existing element if($this->id){ // get values (one-to-many) /** @var \Doctrine\Common\Collections\Collection $values */ $values = $this->getElementValues(); // reset id $this->id = null; // reset values $this->elementValues = new \Doctrine\Common\Collections\ArrayCollection(); // if we had values if(!$values->isEmpty()){ foreach ($values as $value) { // clone it $clonedValue = clone $value; // add to collection $this->addElementValues($clonedValue); } } } } /** * addElementValues * * @param \YourBundle\Entity\ElementValue $elementValue * @return Element */ public function addElementValues(\YourBundle\Entity\ElementValue $elementValue) { if (!$this->getElementValues()->contains($elementValue)) { $this->elementValues[] = $elementValue; $elementValue->setElement($this); } return $this; } с /** * Clone element with values */ public function __clone(){ // we gonna clone existing element if($this->id){ // get values (one-to-many) /** @var \Doctrine\Common\Collections\Collection $values */ $values = $this->getElementValues(); // reset id $this->id = null; // reset values $this->elementValues = new \Doctrine\Common\Collections\ArrayCollection(); // if we had values if(!$values->isEmpty()){ foreach ($values as $value) { // clone it $clonedValue = clone $value; // add to collection $this->addElementValues($clonedValue); } } } } /** * addElementValues * * @param \YourBundle\Entity\ElementValue $elementValue * @return Element */ public function addElementValues(\YourBundle\Entity\ElementValue $elementValue) { if (!$this->getElementValues()->contains($elementValue)) { $this->elementValues[] = $elementValue; $elementValue->setElement($this); } return $this; } 

Где-то просто клонировать его:

 // Returns \YourBundle\Entity\Element which we wants to clone $clonedEntity = clone $this->getElement(); // Do this to say doctrine that we have new object $this->em->persist($clonedEntity); // flush it to base $this->em->flush(); 

Я делаю это:

 if ($form->isValid()) { foreach($classroom->getPupils() as $pupil) { $pupil->setClassroom($classroom); } $em->persist($classroom); $em->flush(); }