使用Spring Data Jpa的CriteriaQuery一个陷阱

网友投稿 313 2022-11-13


使用Spring Data Jpa的CriteriaQuery一个陷阱

使用Spring Data Jpa的CriteriaQuery进行动态条件查询时,可能会遇到一个陷阱,当条件为空时,查询不到任何结果,并不是期望的返回所有结果。这是为什么呢?

例如下述代码,当predicates为空时,返回结果总是为空。

public Page listVmhostSpecWithRelationByPage(String name) {

Specification spec = (root, cq, cb) -> {

root.join("user", JoinType.LEFT);

root.join("tenant", JoinType.LEFT);

List predicates = new ArrayList<>();

......

return cb.or(predicates.toArray(new javax.persistence.criteria.Predicate[0]));

};

PageRequest pagable = PageRequest.of(0, 5);

Page page = vmhostSpecWithRelationDao.findAll(spec, pagable);

return page;

}

看下or的注释就明白了,因为空条件总是为false,而and的空条件总是为true。所以,如果最后是and就没有问题,只有or的时候有问题。

public interface CriteriaBuilder {

/**

* Create a conjunction of the given restriction predicates.

* A conjunction of zero predicates is true.

* @param restrictions zero or more restriction predicates

* @return and predicate

*/

Predicate and(Predicate... restrictions);

/**

* Create a disjunction of the given restriction predicates.

* A disjunction of zero predicates is false.

* @param restrictions zero or more restriction predicates

* @return or predicate

*/

Predicate or(Predicate... restrictions);

}

所以正确的写法应该这样:

public Page listVmhostSpecWithRelationByPage(String name) {

Specification spec = (root, cq, cb) -> {

root.join("user", JoinType.LEFT);

root.join("tenant", JoinType.LEFT);

List predicates = new ArrayList<>();

......

return predicates.isEmpty() ? cb.conjunction() : cb.or(predicates.toArray(new javax.persistence.criteria.Predicate[0]));

};

PageRequest pagable = PageRequest.of(0, 5);

Page page = vmhostSpecWithRelationDao.findAll(spec, pagable);

return page;

}

如果条件为空则返回一个空conjunction,也就是空的and,总是为true。

公司项目的代码中常见这种写法:

public Page listVmhostSpecWithRelationByPage(String name) {

Specification spec = (root, cq, cb) -> {

root.join("user", JoinType.LEFT);

root.join("tenant", JoinType.LEFT);

List predicates = new ArrayList<>();

......

if (predicates.isEmpty()) {

cq.where();

} else {

cq.where(cb.or(predicates.toArray(new javax.persistence.criteria.Predicate[0])));

}

return cq.getRestriction();

};

PageRequest pagable = PageRequest.of(0, 5);

Page page = vmhostSpecWithRelationDao.findAll(spec, pagable);

return page;

}

也能正常工作,但是其实没有必要在toPredicate方法中调用where,toPredicate只需要返回条件,外层会调用where。

public interface Specification extends Serializable {

/**

* Creates a WHERE clause for a query of the referenced entity in form of a {@link Predicate} for the given

* {@link Root} and {@link CriteriaQuery}.

*

* @param root must not be {@literal null}.

* @param query must not be {@literal null}.

* @param criteriaBuilder must not be {@literal null}.

* @return a {@link Predicate}, may be {@literal null}.

*/

@Nullable

Predicate toPredicate(Root root, CriteriaQuNAhSXery> query, CriteriaBuilder criteriaBuilder);

}

本文链接: http://zhongpan.tech/2020/07/20/035-a-trap-for-using-criteriaquery/

以上就是CriteriaQuery使用的一个陷阱的详细内容,更多关于CriteriaQuery 陷阱的资料请关注我们其它相关文章!


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:聊一聊带智能提示的spring
下一篇:api网关技术选型(api网关性能比较)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~