我有两张桌子。
人和他的假期计划。对于这个问题,我们需要知道的是Person包含Id和vp包含此id作为外键,以及年份和开始日期结束日期。人可以包含更多的VPN。 从而
Person VacPlan
_______ ____________
|id |----->|pers_id |
|year |
|start date |
|end date |
我需要选择每个在指定年份的vp中没有单一记录的人。
类似于:select from person where (joined vp where vp.year = :year) is not empty
你有任何想法如何在JPA(eclipse链接)中做到这一点
修改 这是标准的样子
List<Predicate> predicates = new ArrayList<>();
List<Predicate> subPredicates = new ArrayList<>();
CriteriaBuilder cb = getEntityManager().getCriteriaBuilder();
CriteriaQuery<PersonVP> q = cb.createQuery(PersonVP.class);
Root<PersonVP> person = q.from(PersonVP.class);
Subquery<VacationPlanning> subquery = q.subquery(VacationPlanning.class);
Root<VacationPlanning> subRoot = subquery.from(VacationPlanning.class);
Expression<String> exp = person.get("personnelNumber");
subPredicates.add(cb.equal(subRoot.get("year"), year));
subPredicates.add(cb.equal(exp, subRoot.get("perNr")));
subquery.where(cb.and(subPredicates.toArray(new Predicate[subPredicates.size()])));
if (STATUS.FILLED.equals(searchFillStatus)) {
predicates.add(cb.and(cb.exists(subquery)));
} else if (STATUS.NOT_FILLED.equals(searchFillStatus)) {
predicates.add(cb.and(cb.not(cb.exists(subquery))));
}
q.where(cb.and(predicates.toArray(new Predicate[predicates.size()])));
q.select(person);
TypedQuery<PersonVP> query = getEntityManager().createQuery(q);
return query.setFirstResult(from)
.setMaxResults(to)
.getResultList();
答案 0 :(得分:1)
这样的事情应该有效:
select p from Person p where not exists(
select vp.id from Person p2
join p2.vacPlans vp
where p2.id = p.id
and vp.year = :year)
答案 1 :(得分:1)
尝试一下:
SELECT p.id
FROM person p
WHERE p.id NOT IN (SELECT DISTINCT(vp.pers_id)
FROM VacPlan vp)
答案 2 :(得分:1)
您可以使用 NamedQuery 执行此作业。 您在Person Entity文件中使用@JB Nizet查询声明 NamedQuery ,如下所示:
@NamedQuery(name="VpForYear", query="select p from Person p where not exists(
select vp.id from Person p2
join p2.vacPlans vp
where p2.id = p.id
and vp.year = :year)")
你可以像这样使用NamedQuery
:
Query query = em.createNamedQuery("VpForYear");
query.setParameter("year", 2016);