Spring 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);

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

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

你可能感兴趣的:(Spring JPA使用CriteriaBuilder动态构造查询方式)