原则查询“哪里不”子查询问题


Doctrine query "where notIn" subquery issue

在Symfony2中,我在用户和角色之间有很多:很多关系。我正在尝试获取未链接到ROLE_SUPER_ADMIN角色的所有用户的列表。

在迁移到 Symfony2/Doctrine 之前,我已经通过一个简单的 NOT IN sql 查询实现了这一目标,但就我而言,我无法达到与 doctrine 相同的效果。

这是我正在尝试的:

$em = $this->getDoctrine()->getManager();
$qb = $em->createQueryBuilder();
$qb2 = $qb;
$dql = $qb->select('sa.id')
          ->from('AcmeAdminBundle:User', 'sa')
          ->leftJoin('sa.roles', 'r')
          ->andWhere('r.role = :role')
          ->getDQL();
$result = $qb2->select('u')
              ->from('AcmeAdminBundle:User', 'u')
              ->where($qb2->expr()->notIn('u.id', $dql))
              ->setParameter('role', 'ROLE_SUPER_ADMIN')
$users = $result->getQuery()->getResult();

但这是错误:

[Semantical Error] line 0, col 140 near 'sa LEFT JOIN': 
Error: 'sa' is already defined.

这是输出:

SELECT u 
FROM AcmeAdminBundle:User sa 
LEFT JOIN sa.roles r, AcmeAdminBundle:User u 
WHERE u.id NOT IN (
                   SELECT sa.id 
                   FROM AcmeAdminBundle:User sa 
                   LEFT JOIN sa.roles r 
                   WHERE r.role = :role
                  )

不知道为什么它会这样输出,因为它不应该执行两次LEFT JOIN,我怀疑它与拥有两个 QueryBuilder 实例有关,但可能完全是另一回事。

您需要MEMBER OF,或者在您的情况下NOT MEMBER OF选项。

$qb->select('sa.id')
   ->from('AcmeAdminBundle:User', 'sa')
   ->where(":Role NOT MEMBER OF sa.roles")
   ->setParameter("Role", <<ROLE_ID_OR_ROLE_ENTITY>>);

我没有测试这段代码,但它应该给你一些想法。

完整的文档可以在 https://www.doctrine-project.org/projects/doctrine-orm/en/2.14/reference/dql-doctrine-query-language.html