Я пытаюсь создать коррелированный запрос, используя criteria
в спящем режиме.
Ниже приведен код:
ожидаемый SQL-запрос:
select * from
COLLEGE c
where (select count(s.college_id) from student s where c.college_id = s.college_id group by s.college_id)> 2;
, но он генерирует, есть NO join условие между внешним колледжем ( колледж0_ ) и вспомогательным колледжем ( колледж2_ )
SELECT
college0_.COLLEGE_ID AS COLLEGE_1_2_,
college0_.NAME AS NAME2_2_
FROM
COLLEGE college0_
WHERE
(SELECT
COUNT(student1_.COLLEGE_ID)
FROM
STUDENT student1_
INNER JOIN
COLLEGE college2_ ON student1_.COLLEGE_ID = college2_.COLLEGE_ID
GROUP BY college2_.COLLEGE_ID) >= 3
Способ выполнения аналогичным образом:
public List<College> findCollegesWithMoreThanXStudentsUsingCriteriaJpa(long x){
Session session = getSession();
CriteriaBuilder cb = session.getCriteriaBuilder();
String hql = " from College c where (select count(s.college) from Student s join s.college c1 where c.id = c1.id group by s.college) > :cnt"; // I am trying the same query using criteria
CriteriaQuery<College> collegeCriteriaQuery = cb.createQuery(College.class);
Root<College> collegeRoot = collegeCriteriaQuery.from(College.class);
collegeCriteriaQuery.select(collegeRoot);
Subquery<Long> subquery = collegeCriteriaQuery.subquery(Long.class);
Root<Student> studentRoot = subquery.from(Student.class);
subquery.select(cb.count(studentRoot.get("college")));
Join<College,Student> studentCollegeJoin = studentRoot.join("college");
subquery.correlate(collegeRoot);
subquery.groupBy(studentCollegeJoin.get("id"));
collegeCriteriaQuery.where(cb.greaterThanOrEqualTo(subquery,x));
return session.createQuery(collegeCriteriaQuery).getResultList();
}
классы сущностей:
Student.java
import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import java.util.List;
@Entity
@Table(name = "STUDENT")
public class Student {
@Id
@Column(name = "STUDENT_ID")
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int id;
@Column(name="NAME")
private String name;
@ManyToOne
@JoinColumn(name = "COLLEGE_ID",referencedColumnName = "COLLEGE_ID")
private College college;
@OneToMany(cascade = {CascadeType.ALL})
@JoinColumn(name = "STUDENT_ID",referencedColumnName = "STUDENT_ID")
private List<StudentAddress> studentAddresses;
public int getId() {
return id;
}
public Student setId(int id) {
this.id = id;
return this;
}
public String getName() {
return name;
}
public Student setName(String name) {
this.name = name;
return this;
}
public College getCollege() {
return college;
}
public Student setCollege(College college) {
this.college = college;
return this;
}
public List<StudentAddress> getStudentAddresses() {
return studentAddresses;
}
public Student setStudentAddresses(List<StudentAddress> studentAddresses) {
this.studentAddresses = studentAddresses;
return this;
}
@Override
public String toString() {
return "Student{" +
"id=" + id +
", name='" + name + '\'' +
", college=" + college +
", studentAddresses=" + studentAddresses +
'}';
}
}
College.java
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
@Entity
@Table(name = "COLLEGE")
public class College {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "COLLEGE_ID")
private int id;
@Column(name = "NAME")
private String name;
public int getId() {
return id;
}
public College setId(int id) {
this.id = id;
return this;
}
public String getName() {
return name;
}
public College setName(String name) {
this.name = name;
return this;
}
@Override
public String toString() {
return "College{" +
"id=" + id +
", name='" + name + '\'' +
'}';
}
}