Spring JPA使用CriteriaBuilder动态构造查询方式

网友投稿 803 2022-09-10


Spring JPA使用CriteriaBuilder动态构造查询方式

目录Spring JPA使用CriteriaBuilder动态构造查询JPA CriteriaBuilder中一些运算的使用

Spring JPA使用CriteriaBuilder动态构造查询

在使用Spring JPA提供的方法只能进行简单的CRUD,如果遇到复杂的情况就需要我们动态来构建查询条件了。这里我们来看使用CriteriaBuilder如何来构造查询。

核心代码:

CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();

CriteriaQuery query = criteriaBuilder.createQuery(Long.class);

Root root = query.from(Order.class);

query.select(criteriaBuilder.count(root.get("id")));

Predicate predicate = criteriaBuilder.equal(root.get("id"), 1);

query.where(predicate);

Long singleResult = entityManager.createQuery(query).getSingleResult();

调用entityManager.getCriteriaBuilder()来获取CriteriaBuilder。CriteriaBuilder可以用于创建CriteriaQuery、CriteriaUpdate和CriteriaDelete。除此之外类似count、max等函数也是由CriteriaBuilder来创建的。其中Entitymanager可以使用@PersistenceContext注解来进行注入。

调用criteriaBuilder.createQuery来创建CriteriaQuery。其中createQuery的参数是Query返回值类型。

调用query.from(Order.class)。参数是对应于order表的实体类,query.from类似于sql中的from语句,该方法的执行等价于sql中的from order。

调用 query.select创建映射。 query.select(criteriaBuilder.count(root.get(“id”)))等价于select count(id)。如果执行query.select(root)则等价于select *。

使用CriteriaBuilder构造查询条件Predicate,该predicate也就是在where后面的条件子句。

将Predicate放在 query.where中。

最后执行查询获取数据。

JPA CriteriaBuilder中一些运算的使用

最近使用jpa时,需要使用订单中的金额除以单价算出每个订单的数量,然后求和。找了好多资料才解决,在此整理一下。

首先了解一下CriteriaBuilder的一些运算

// Create path and parameter expressions:

Expression path = country.get("population");

Expression param = cb.parameter(Integer.class);

// Addition (+)

Expression sum1 = cb.sum(path, param); // expression + expression

Expression sum2 = cb.sum(path, 1000); // expression + number

Expression sum3 = cb.sum(1000, path); // number + expression

// Subtraction (-)

Expression diff1 = cb.diff(path, param); // expression - expression

Expression diff2 = cb.diff(path, 1000); // expression - number

Expression diff3 = cb.diff(1000, path); // number - expression

// Multiplication (*)

Expression prod1 = cb.prod(path, param); // expression * expression

Expression prod2 = cb.prod(path, 1000); // expression * number

Expression prod3 = cb.prod(1000, path); // number * expression

// Division (/)

Expression quot1 = cb.quot(path, param); // expression / expression

Expression quot2 = cb.quot(path, 1000); // expression / number

Expression quot3 = cb.quot(1000, path); // number / expression

// Modulo (%)

Expression mod1 = cb.mod(path, param); // expression % expression

Expression mod2 = cb.mod(path, 1000); // expression % number

Expression mod3 = cb.mod(1000, path); // number % expression

// Math(abs, exp, sqrt)

Expression abs = cb.abs(param); // 求绝对值ABS(expression)

Expression neg = cb.neg(path); // 求相反数 -expression

Expression sqrt = cb.sqrt(cb.literal(100)); //求平方根 SQRT(expression)

由于CriteriaBuilder提供的加减乘除方法的名字和平常使用的不太一样,所以用了好久才找出来。

单字段求和可以直接使用

CriteriaBuilder cb = em.getCriteriaBuilder();

Expression sum = cb.sum(root.get(字段名)).alias(别名)

前边也说了需求是用金额(amount)除以单价(unitPrice),然后求和,所以这时需要先用amount除以unitPrice

Expression quot = cb.quot(root.get("amount"), root.get("unitPrice"));

算出数量后就可以使用sum求和了。

cb.sum(quot)

完事之后,还需要对数据四舍五入,需要用到mysql的round方法

在CriteriaBuilder没有找到round方法,那怎么办呢?没关系,CriteriaBuilder还提供了function方法,在function方法里可以直接传方法名进去

/**

name: 方法名

returnType: 返回类型

arguments:表达式

**/

public Expression function(String name, Class returnType, Expression... arguments) {

return new ParameterizedFunctionExpression(this, returnType, name, arguments);

}

调用代码如下

Expression round = cb.function("round", BigDecimal.class, quot);

到这一步发现,还需要保留两位小数。可是,Expression里 没有相关的方法。天无绝人之路,在看了cb.function()后发现,这个方法的最后一个参数是可变参数,所以上边的代码改成了如下

List list = new ArrayList<>();

list.add(1);

list.add(2);

Expression size = cb.size(list);

Expression round = cb.function("round", BigDecimal.class, quot,size);

至此,完成了先求两列的商,然后对商求和的功能。


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

上一篇:DDoS 保护、缓解和防御:8 个基本技巧(ddos攻击判刑多久)
下一篇:DDoS 保护、缓解和防御:8 个基本技巧(3~4)(ddos在线攻击平台免费)
相关文章

 发表评论

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