QueryDSL-如何加入子查询的并集
我正在使用QueryDSL构建SQL查询,该查询包含以联合身份加入的多个子查询。这是我查询的基础:
QTransaction t = QTransaction.transaction;query = query.from(t).where(t.total.gt(BigDecimal.ZERO));
然后,我有几个子查询来获取与事务关联的客户端名称。我将示例缩减为两个:
SQLSubQuery subQuery = new SQLSubQuery();subQuery = subQuery.from(t).join(t.fk462bdfe3e03a52d4, QClient.client);
ListSubQuery clientByPaid = subQuery.list(t.id, bt.paidId, QClient.client.name.as("clientname"));
subQuery = new SQLSubQuery();
subQuery = subQuery.from(t).where(t.paidId.isNull(), t.clientname.isNotNull());
ListSubQuery clientByName = subQuery.list(t.id, Expressions.constant(-1L), t.clientname.as("clientname"));
如何将它们结合在一起,并通过我的主要查询加入结合?这是我目前的尝试:
subQuery = new SQLSubQuery();subQuery = subQuery.from(subQuery.unionAll(clientByPaid,clientByName).as("namequery"));
query = query.leftJoin(subQuery.list(
t.id, Expressions.path(Long.class, "clientid"),
Expressions.stringPath("clientname")),
Expressions.path(List.class, "namequery"));
这样可以编译,但是在尝试运行时会在运行时生成无效的SQL query.count()
。可能的错误:
- 子查询联合的语法。
.as(...)
命名子查询结果列的表达式与中使用的路径表达式之间的连接leftJoin
。
回答:
解决它。主要错误是我错过了on
左联接中的子句,但是为了表达on
条件,我在命名子查询时必须更加小心。该文档略微说明了构造访问子查询结果的路径,因此这里是示例。
联合中的第一个查询设置列名:
SQLSubQuery subQuery = new SQLSubQuery();subQuery = subQuery.from(t).join(t.fk462bdfe3e03a52d4, QClient.client);
ListSubQuery clientByPaid = subQuery.list(t.id.as("id"), t.paidId.as("clientid"),
QClient.client.name.as("clientname"));
subQuery = new SQLSubQuery();
subQuery = subQuery.from(t).where(t.paidId.isNull(), t.clientname.isNotNull());
ListSubQuery clientByName = subQuery.list(t.id, Expressions.constant(-1L),
t.clientname);
现在,我需要构建一个路径表达式以引用回我的内部查询。我使用哪个类似乎并不重要,所以我选择了Void来强调这一点。
subQuery = new SQLSubQuery();Path innerUnion = Expressions.path(Void.class, "innernamequery");
subQuery = subQuery.from(subQuery.union(clientByPaid,clientByName).as(innerUnion));
并进一步表达该on
子句的路径表达。请注意,我加入list()
了联合查询的一个,并使用innerUnion
前面定义的路径选择了每一列。
Path namequery = Expressions.path(Void.class, "namequery");query = query.leftJoin(subQuery.list(
Expressions.path(Long.class, innerUnion, "id"),
Expressions.path(Long.class, innerUnion, "clientid"),
Expressions.stringPath(innerUnion, "clientname")),
namequery)
.on(t.id.eq(Expressions.path(Long.class, namequery, "id")));
以上是 QueryDSL-如何加入子查询的并集 的全部内容, 来源链接: utcz.com/qa/426545.html