Hibernate中带有不存在的Criteria API查询

3

我正在尝试编写查询,以返回未分配到路线的司机列表。

我的数据库设置如下。

Route:
route_id
user_id//specified as driver

User:
user_id
role // need to select user, which is Driver role

只有路线可以看到用户(司机),用户(司机)看不到路线。

这是我尝试编写此类查询的方法。

public List<User> getUnsignedDrivers(){
    CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();

    CriteriaQuery<User> query = criteriaBuilder.createQuery(User.class);
    Root<User> user = query.from(User.class);
    query.select(user);

    Subquery<Route> subquery = query.subquery(Route.class);
    Root<Route> subRootEntity = subquery.from(Route.class);
    Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get("Route_.User"), user);
    subquery.where(correlatePredicate);
    query.where(criteriaBuilder.not(criteriaBuilder.exists(subquery)));

    TypedQuery<User> typedQuery = entityManager.createQuery(query);
    return typedQuery.getResultList();
}

我是JPA的新手,所以这是我的问题。

更具体地说,我需要选择未设置到任何路线的司机角色用户。

我的实体设置如下:

    @Entity
public class Route {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;

@OneToOne(fetch = FetchType.EAGER, cascade = {})
@JoinColumn(name = "user_id", nullable = true)
private User driver;
.....
@Entity
public class User {

public static enum Role {
    ADMIN, MANAGER, DRIVER;
}

@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;

@Enumerated(EnumType.STRING)
@Column(nullable = false)
private Role role;

更新: 当前的查询

CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();

    CriteriaQuery<User> query = criteriaBuilder.createQuery(User.class);
    Root<User> user = query.from(User.class);
    Predicate predicateRole = criteriaBuilder.equal(user.get("role"), User.Role.DRIVER);
    query.where(predicateRole);
    query.select(user);


    Subquery<Route> subquery = query.subquery(Route.class);
    Root<Route> subRootEntity = subquery.from(Route.class);
    Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get("driver"), user);
    subquery.where(correlatePredicate);
    query.where(criteriaBuilder.not(criteriaBuilder.exists(subquery)));

    TypedQuery<User> typedQuery = entityManager.createQuery(query);
    return typedQuery.getResultList();

问题仍然存在。
我得到了这个异常:
java.lang.IllegalStateException: No explicit selection and an implicit one cold not be determined
at org.hibernate.ejb.criteria.QueryStructure.locateImplicitSelection(QueryStructure.java:296)
at org.hibernate.ejb.criteria.QueryStructure.render(QueryStructure.java:249)
at org.hibernate.ejb.criteria.CriteriaSubqueryImpl.render(CriteriaSubqueryImpl.java:282)
at org.hibernate.ejb.criteria.predicate.ExistsPredicate.render(ExistsPredicate.java:58)
at org.hibernate.ejb.criteria.QueryStructure.render(QueryStructure.java:258)
at org.hibernate.ejb.criteria.CriteriaQueryImpl.render(CriteriaQueryImpl.java:340)
at org.hibernate.ejb.criteria.CriteriaQueryCompiler.compile(CriteriaQueryCompiler.java:217)
at org.hibernate.ejb.AbstractEntityManagerImpl.createQuery(AbstractEntityManagerImpl.java:587)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:240)
at $Proxy25.createQuery(Unknown Source)

在这一行代码中,throws指的是抛出异常。 TypedQuery<User> typedQuery = entityManager.createQuery(query); 这段代码会创建一个类型为TypedQuery的对象,并执行查询操作。

解决方法 以下方法对我来说非常有效。 我写了这个解决方案,因为我不能使用反向关系。

public List<User> getUnsignedDrivers(){
    CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();

    CriteriaQuery<User> query = criteriaBuilder.createQuery(User.class);
    Root<User> user = query.from(User.class);
    Predicate predicateRole = criteriaBuilder.equal(user.get("role"), User.Role.DRIVER);
    query.where(predicateRole);
    query.select(user);

    TypedQuery<User> typedQuery = entityManager.createQuery(query);
    List<User> allDrivers = typedQuery.getResultList();
    List<User> notAssignedDrivers = new ArrayList<User>();
    List<Route> haveDriverRoutes = getRouteWithNoDrives();
    for (User driver : allDrivers){
        if (!isDriverAssigned(haveDriverRoutes,driver.getId())){
            notAssignedDrivers.add(driver);
        }
    }
    return notAssignedDrivers;
}

private boolean isDriverAssigned(List<Route> haveDriverRoutes, long driverId){
    for(Route route : haveDriverRoutes){
        if (route.getDriver().getId() == driverId){
            return true;
        }
    }
    return false;
}

@SuppressWarnings("unchecked")
public List<Route> getRouteWithNoDrives() {
    Query query = entityManager.createQuery("SELECT o FROM " + type.getSimpleName() + " o WHERE o.driver != null");
    return  query.getResultList();
}
2个回答

1

您在User实体中缺少反向的OneToOne关系:

@OneToOne(mappedBy="driver")
private Route route;

请参考此链接了解如何映射一对一关系。

在这部分中,您有一个错误:subRootEntity.get("Route_.User")。这不是有效的语法,并且在Route实体中没有名为User的属性:根据您最新的编辑,该属性称为driver

您有两种方法可以获取路径表达式,要么使用:

Path<User> path = subRootEntity.get("driver");
// in a compact way: 
Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get("driver"), user);

或通过元模型的使用:

Path<User> path = subRootEntity.get(Route_.driver);
// in a compact way: 
Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get(Route_.driver), user);

您似乎混淆了这两种方法。请参阅文章以获取有关使用元模型的更多信息。

查询的其余部分看起来是正确的。


1
感谢您的帖子。是的,我在(subRootEntity.get("Route_.User"), user)中出现了错误,这是因为我不知道如何定义正确的路径以匹配criteriaBuilder.equal(subRootEntity.get("Route_.User"), user) 中的 User。我尝试使用Path<User> path = subRootEntity.get("user"),正如您所说的那样,但它也抛出了一个异常,指出 "user" 路径不存在。我该如何获取正确的路径? - kuldarim
subRootEntity.get("driver"); 这个路径是可行的,现在我改成了 Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get("driver").get("id"), user.get("id")); 但是它抛出了 java.lang.IllegalStateException: No explicit selection and an implicit one cold not be determinedat org.hibernate.ejb.criteria 异常。有什么建议吗? - kuldarim
我也尝试了 Predicate correlatePredicate = criteriaBuilder.equal(subRootEntity.get("driver"), user); 但是它抛出了 java.lang.IllegalStateException: No explicit selection and an implicit one cold not be determined 异常。 - kuldarim
我需要实现单向的 @OneToOne,因为司机不应该关心路线,只有路线应该知道司机。司机是用户,用户有3个角色:经理、司机和管理员,所以如果必须使用反向的 @OneToOne,那将会很遗憾。 - kuldarim
查询是否与反向关系一起工作?是的。没有反向关系,它能否工作?不行。您读过我在答案中提供的链接吗?它说这是必需的。那么问题出在哪里呢? - perissf

1

你添加子查询的那一行应该返回一些内容。因此,你需要将代码更改为:

query.where(criteriaBuilder.exists(subquery).not());
query.where(criteriaBuilder.not(criteriaBuilder.exists(subquery.select(subRootEntity))));

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接