Подтвердить что ты не робот

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');
}

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

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

4b9b3361

Ответ 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();
}