Symfony2:从集合中删除实体

noS*_*owP 5 php symfony doctrine-orm

1.概述

我希望能够使用symfony2表单从集合中删除实体.

1.1问题

我可以向集合中添加新实体,并将其删除,只要添加或删除的实体位于集合末尾即可.只要我从开头或中间删除一个,我就会收到以下错误:

当我尝试这样做时,我收到此错误:

属性"id"和方法"addId()"/"removeId()","setId()","id()","__ set()"或"__call()"之一都不存在并且公开在"ApiBundle\Entity\Data\Column"类中访问.

1.2代码

这是所有相关的代码.

数据

/**
 * Data
 *
 * @ORM\Table(name="data__data")
 * @ORM\Entity(repositoryClass="ApiBundle\Repository\Data\DataRepository")
 */
class Data
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="string")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="UUID")
     */
    protected $id;

    /**
     * @var ArrayCollection
     * @ORM\OneToMany(targetEntity="Column", mappedBy="parent", cascade={"all"}, orphanRemoval=true)
     */
    protected $columns;

    /**
     * Initialise the array collections
     */
    public function __construct()
    {
        $this->columns = new ArrayCollection();
    }

    /**
     * @param mixed $columns
     */
    public function setColumns($columns)
    {
        $this->columns = $columns;
    }

    /**
     * @param Column $column
     */
    public function addColumn($column)
    {
        $column->setParent($this);
        $this->columns->add($column);
    }

    /**
     * @param Column $column
     */
    public function removeColumn($column)
    {
        $this->columns->removeElement($column);
    }
}
Run Code Online (Sandbox Code Playgroud)

/**
 * Data
 *
 * @ORM\Table(name="data__column")
 * @ORM\Entity
 */
class Column
{

    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="string")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="UUID")
     */
    protected $id;

    /**
     * @var Data
     * @ORM\ManyToOne(targetEntity="Data", inversedBy="columns")
     */
    protected $parent;

    /**
     * @return Data
     */
    public function getParent()
    {
        return $this->parent;
    }

    /**
     * @param Data $parent
     */
    public function setParent($parent)
    {
        $this->parent = $parent;
    }
}
Run Code Online (Sandbox Code Playgroud)

DataFormType

class DataFormType extends AbstractType {

    public function buildForm(FormBuilderInterface $builder, array $options)
    {

        $builder
            ->add('id')
            ->add('columns', 'collection', array(
                'type' => new ColumnFormType(),
                'allow_add'    => true,
                'allow_delete' => true,
                'by_reference' => false
            ))
        ;
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'ApiBundle\Entity\Data\Data',
            'csrf_protection' => false
        ));
    }

    public function getName()
    {
        return 'data';
    }

} 
Run Code Online (Sandbox Code Playgroud)

ColumnFormType

class ColumnFormType extends AbstractType
{

    public function buildForm(FormBuilderInterface $builder, array $options)
    {

        $builder
            ->add('id');
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'ApiBundle\Entity\Data\Column',
            'csrf_protection' => false
        ));
    }

    public function getName()
    {
        return 'data_column';
    }

} 
Run Code Online (Sandbox Code Playgroud)

为清楚起见,我已从这些代码段中删除了一些代码

1.3结论

就像我说的那样,从集合的末尾添加或删除时我没有遇到任何问题.但是,只要它在任何其他地方,它就会出错.

谢谢你的帮助.

ori*_*nal 2

该错误是由于缺少集合密钥保存而导致的。

CollectionType是高坚韧的ResizeListener。它用子表单填充集合表单:

public function preSetData(FormEvent $event)
{
    $form = $event->getForm();
    $data = $event->getData();

    ...

    // Then add all rows again in the correct order
    foreach ($data as $name => $value) {
        $form->add($name, $this->type, array_replace(array(
            'property_path' => '['.$name.']',
        ), $this->options));
    }
}
Run Code Online (Sandbox Code Playgroud)

因此,每个子表单都映射到集合对象(底层数据),并具有适用于集合索引的名称,例如“[0]”、“[1]”。当您从集合中删除元素时,ResizeListener会删除多余的子表单。

public function preSubmit(FormEvent $event)
{
    $form = $event->getForm();
    $data = $event->getData();
    ...

    // Remove all empty rows
    if ($this->allowDelete) {
        foreach ($form as $name => $child) {
            if (!isset($data[$name])) {
                $form->remove($name);
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

可以说有data[columns][0][id]=1, data[columns][1][id]=2, data[columns][2][id]=3

当您从末尾删除一个元素时 - 一切都很好。都有data[columns][0][id]=1, data[columns][1][id]=2相应的内容。然后子表单[2]将被删除,然后索引为2的元素将从集合中删除。

当您删除不在末尾的元素并且不保留键时 - 就会发生错误。例如你发送data[columns][0][id]=2, data[columns][1][id]=3. ResizeListener将删除索引为的子表单[2]。其余子表单 ( [0][1]) 及其子表单 ( id) 的基础数据将被覆盖。大多数嵌套子表单都会首先处理。

 [0] (Column)
    [id]
        1 => 2
 [1] (Column)
    [id]
        2 => 3
Run Code Online (Sandbox Code Playgroud)

然后PropertyPathMapper将检测id子表单的数据不等于Column的id属性值(这是底层数据[0]):

public function mapFormsToData($forms, &$data)
{
    ...
            if (!is_object($data) || !$config->getByReference() || $form->getData() !== $this->propertyAccessor->getValue($data, $propertyPath)) {
                $this->propertyAccessor->setValue($data, $propertyPath, $form->getData());
            }
    ...
}
Run Code Online (Sandbox Code Playgroud)

它将为对象PropertyAccessor设置新idColumn。最后一个将抛出异常,因为无法id为 Column 设置 new (没有 setter,属性不是公共的,等等)。

解决方案:保留按键顺序。如果你得到data[columns][0][id]=1, data[columns][1][id]=2, data[columns][2][id]=3并删除了你应该发送的第一个元素data[columns][1][id]=2, data[columns][2][id]=3

PS保留表单的键顺序对于所有情况都是良好的做法。它将防止您进行多余的UPDATE查询和循环。