2

我在两个实体之间有一个正常的多对多关系:用户和对象。

________        _________________        ____________
| User |        |   User_Object |        |  Object  |
|------|        |---------------|        |----------|
|  id  |        |     user_id   |        |    id    |
| .... |        |    object_id  |        |   ....   |
|______|        |_______________|        |__________|

我想批量删除相当大的用户集(以及在 User_Object 表中与它们关联的记录)。逐个删除实体不足以满足我的需求(对于 > 1000 个实体,它需要很长时间)。

//This method is far too slow for my needs
$qb = $this->doctrine->em->createQueryBuilder();
$qb->select('u')
    ->from('Entities\User', 'u')
    ->where("u.whatever= ?1")
    ->setParameter(1, $whatever);
$users = $qb->getQuery()->getResult();

foreach($users as $user)
{
    $this->doctrine->em->remove($user);
    //...

学说文档说,批量删除实体最有效的是 DQL,它会给我类似的东西:

$qb = $this->doctrine->em->createQuery('delete from Entities\User u where u.whatever = ?1');
$qb->setParameter(1, $whatever);
$numDeleted = $qb->execute(); //This will throw because of User_Object records

由于 User_Object 连接表中的记录,这将引发异常(参照完整性异常)。

所以,我的问题是:如何在批量删除场景中有效地删除连接表中的记录。

我真的很想避免向它抛出原始 SQL,我的其余代码在任何地方都使用实体,如果可能的话,我希望保持这种方式。

编辑:关系被标记为这样(我使用 yml):

manyToMany:
  objects:
  targetEntity: Object
  inversedBy: users
  cascade: ["remove"]
  joinTable:
    name: User_Object
    joinColumns:
      user_id:
        referencedColumnName: id
    inverseJoinColumns:
      object_id:
        referencedColumnName: id
4

1 回答 1

0

如果要删除对象的用户,则尚未设置教义的行为。您必须说如果用户被删除,那么所有相关对象也必须被删除。

/**
 * @ORM\ManyToMany(targetEntity="User", mappedBy="objects", cascade={"remove"})
 */
protected $users;


/**
 * @ORM\ManyToMany(targetEntity="Object", inversedBy="users", cascade={"persist", "remove"})
 * @ORM\JoinTable(name="user_object",
 * joinColumns={@ORM\JoinColumn(name="user_id", referencedColumnName="id")},
 * inverseJoinColumns={@ORM\JoinColumn(name="object_id", referencedColumnName="id")}
 * )
 */
protected $objects;

这是一个例子。我没有测试它。

于 2013-11-04T16:42:00.297 回答