Symfony2/Doctrine: как повторно сохранить объект с OneToMany в качестве каскадной новой строки
Во-первых, этот вопрос похож на Как повторно сохранить объект как еще одну строку в Doctrine 2
Разница в том, что я пытаюсь сохранить данные в сущности, у которой есть отношения OneToMany. Я хотел бы повторно сохранить объект как новую строку в родительском объекте (на стороне "один" ), а затем как новые строки в каждом последующем дочернем элементе (со стороны "много" ).
Я использовал довольно простой пример класса, в котором было много учеников, чтобы это было просто.
Итак, у меня может быть ClassroomA с id = 1 и у него есть 5 учеников (от 1 до 5). Я хотел бы знать, как я мог, в Doctrine2, взять этот Entity и снова сохранить его в базе данных (после потенциальных изменений данных), все с новыми идентификаторами повсюду, а исходные строки остаются нетронутыми во время persist/flush.
Позволяет сначала определить наши объекты 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 ============
}
Учреждение ученика:
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');
}
Я пробовал использовать клон, но только сохранил родительские отношения (класс в нашем примере) со свежим идентификатором, в то время как данные о детях (ученики) были обновлены по сравнению с исходными идентификаторами.
Заранее благодарим за любую помощь.
Ответы
Ответ 1
Вещь с 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()
(in Classroom
), и вызвать это после того, как вы клонируете объект. В любом случае, вы можете использовать тот же код внутри методов __clone()
или clonePupils()
.
Когда вы клонируете родительский класс, эта функция создаст новую коллекцию, полную клонов дочерних объектов.
$cloneClassroom = clone $classroom;
$cloneClassroom->clonePupils();
$em->persist($cloneClassroom);
$em->flush();
Вероятно, вы захотите, чтобы каскад сохранялся в вашей коллекции $pupils
, чтобы упростить задачу, например
/**
* @ORM\OneToMany(targetEntity="Pupil", mappedBy="classroom", cascade={"persist"})
*/
protected $pupils;
Ответ 2
Я сделал это так, и он отлично работает.
Внутри клонированной сущности мы имеем магию __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;
}
Где-то просто клонировать его:
// 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();
Ответ 3
Я делаю это:
if ($form->isValid()) {
foreach($classroom->getPupils() as $pupil) {
$pupil->setClassroom($classroom);
}
$em->persist($classroom);
$em->flush();
}