gpt4 book ai didi

php - Symfony 表单不保存具有 ManyToMany 关系的实体

转载 作者:可可西里 更新时间:2023-10-31 23:39:34 24 4
gpt4 key购买 nike

我在通过多对多关系的形式保存实体时遇到问题。

我无法保存关系“mappedBy”端的字段。

下面的代码没有将任何内容保存到数据库中,也没有抛出任何错误:

// Entity/Pet
/**
* @var \Doctrine\Common\Collections\Collection
*
* @ORM\ManyToMany(targetEntity="AppBundle\Entity\Customer", mappedBy="pet", cascade={"persist"})
*/
private $customer;

/**
* Set customer
*
* @param \AppBundle\Entity\Customer $customer
* @return Pet
*/
public function setCustomer($customer)
{
$this->customer = $customer;

return $this;
}

// Entity/Customer
/**
* @var Pet
*
* @ORM\ManyToMany(targetEntity="AppBundle\Entity\Pet", inversedBy="customer", cascade={"persist"})
* @ORM\JoinTable(name="customer_pet",
* joinColumns={
* @ORM\JoinColumn(name="customer_id", referencedColumnName="id")
* },
* inverseJoinColumns={
* @ORM\JoinColumn(name="pet_id", referencedColumnName="id")
* }
* )
*/
private $pet;

// PetType.php
$builder->add('customer', 'entity',
array(
'class' => 'AppBundle:Customer',
'property' => 'firstname',
'empty_value' => 'Choose owner',
'multiple' => true
));

它正在以相反的方式运作。因此,如果我从 CustomerType 保存一些东西,一切正常。

编辑:

下面的解决方案对我有用,但几天后我发现该解决方案存在问题。如果提交的表单带有已经保存在数据库中的值,那么 Symfony 将抛出一个错误。为了防止这种情况,我必须检查给定的客户是否已经分配给宠物。

必须在函数开始时检查当前分配的客户,而不是在提交表单之后,因为出于某种原因,在提交之后 Pet() 对象包含提交的值,而不仅仅是那些存在于数据库中的值。

所以一开始我将所有已经分配的客户放入数组

  $em = $this->getDoctrine()->getManager();
$pet = $em->getRepository('AppBundle:Pet')->find($id);
$petOriginalOwners = array();
foreach ($pet->getCustomer() as $petCustomer)
{
$petOriginalOwners[] = $petCustomer->getId();
}

在提交表单后,我检查了提交的 ID 是否在数组中

if ($form->isValid()) 
{
foreach ($form['customer']->getData()->getValues() as $v)
{
$customer = $em->getRepository('AppBundle:Customer')->find($v->getId());
if ($customer && !in_array($v->getId(), $petOriginalOwners) )
{
$customer->addPet($pet);
}
}
$em->persist($pet);
$em->flush();
return $this->redirect($this->generateUrl('path'));
}

最佳答案

在 Symfony2 中,具有 inversedBy 学说注释属性的实体应该编辑由多对多关系创建的额外表。这就是为什么当你创建一个客户时,它会在那个额外的表中插入相应的行,保存相应的宠物。

如果您希望相反的方式发生相同的行为,我建议:

//PetController.php
public function createAction(Request $request) {
$entity = new Pet();
$form = $this->createCreateForm($entity);
$form->submit($request);



if ($form->isValid()) {
$em = $this->getDoctrine()->getManager();
foreach ($form['customer']->getData()->getValues() as $v) {
$customer = $em->getRepository('AppBundle:Customer')->find($v->getId());
if ($customer) {
$customer->addPet($entity);
}
}
$em->persist($entity);
$em->flush();

return $this->redirect($this->generateUrl('pet_show', array('id' => $entity->getId())));
}

return $this->render('AppBundle:pet:new.html.twig', array(
'entity' => $entity,
'form' => $form->createView(),
));
}

private function createCreateForm(Pet $entity) {
$form = $this->createForm(new PetType(), $entity, array(
'action' => $this->generateUrl('pet_create'),
'method' => 'POST',
));

return $form;
}

这两个只是标准的 Symfony2 CRUD 在对应于 Pet 实体的 Controller 中生成的操作。

唯一的调整是在第一个操作中插入的 foreach 结构,这样您就可以强制将相同的宠物添加到您在表单中选择的每个客户,从而获得所需的行为。

看,这很可能不是正确的方法或正确的方法,但它是一种方法并且有效。希望对您有所帮助。

关于php - Symfony 表单不保存具有 ManyToMany 关系的实体,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30463674/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com