如何使用子查询转换SQL查询并计数到JPA条件构建器

时间:2020-12-31 00:10:24

I'm trying to convert this SQL query to use a JPA criteria builder. The answer needs to be a double or a float.

我正在尝试将此SQL查询转换为使用JPA条件构建器。答案需要是双重或浮动。

SELECT CAST((COUNT(m.id) -
(SELECT COUNT(s.id) 
  FROM mobile_unit as s
  left JOIN incident as i ON s.incidentId=i.id
  JOIN organizational_unit as o ON s.organizationalUnitId=o.id
  WHERE (s.organizationalUnitId = 1 AND s.incidentId IS NULL))) AS float) 
/COUNT(m.id) 
FROM mobile_unit as m 
JOIN organizational_unit as o ON m.organizationalUnitId=o.id
WHERE m.organizationalUnitId = 1

1 个解决方案

#1


0  

Without testing this, here is a rough sketch of how this could be done.

如果不对此进行测试,这里是一个如何做到这一点的草图。

CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Float> cq = cb.createQuery(Float.class);

Root<MobileUnit> root = cq.from(MobileUnit.class);
// Join is actually unnecessary
root.join("organizationalUnit", JoinType.INNER);
cq.where(cb.equal(
    root.get("organizationalUnitId"),
    1
));

Subquery<Long> subquery = cq.subquery(Long.class);
Root<MobileUnit> subRoot = subquery.from(MobileUnit.class);
// Actually these joins are unnecessary, but you requested them..
subRoot.join("incident", JoinType.LEFT);
subRoot.join("organizationalUnit", JoinType.INNER);

subquery.select(cb.count(subRoot.get("id")));
subquery.where(cb.and(
    subRoot.get("organizationalUnitId").eq(cb.literal(1)),
    subRoot.get("incidentId").isNull()
));

cq.select(cb.quot(
    cb.diff(
        cb.count(root.get("id")),
        subquery
    ).as(Float.class),
    cb.count(root.get("id"))
));

#1


0  

Without testing this, here is a rough sketch of how this could be done.

如果不对此进行测试,这里是一个如何做到这一点的草图。

CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Float> cq = cb.createQuery(Float.class);

Root<MobileUnit> root = cq.from(MobileUnit.class);
// Join is actually unnecessary
root.join("organizationalUnit", JoinType.INNER);
cq.where(cb.equal(
    root.get("organizationalUnitId"),
    1
));

Subquery<Long> subquery = cq.subquery(Long.class);
Root<MobileUnit> subRoot = subquery.from(MobileUnit.class);
// Actually these joins are unnecessary, but you requested them..
subRoot.join("incident", JoinType.LEFT);
subRoot.join("organizationalUnit", JoinType.INNER);

subquery.select(cb.count(subRoot.get("id")));
subquery.where(cb.and(
    subRoot.get("organizationalUnitId").eq(cb.literal(1)),
    subRoot.get("incidentId").isNull()
));

cq.select(cb.quot(
    cb.diff(
        cb.count(root.get("id")),
        subquery
    ).as(Float.class),
    cb.count(root.get("id"))
));