jpa+mysql分组查询_Spring Data JPA实现动态查询的两种方法

前言

一般在写业务接口的过程中,很有可能需要实现可以动态组合各种查询条件的接口。如果我们根据一种查询条件组合一个方法的做法来写,那么将会有大量方法存在,繁琐,维护起来相当困难。想要实现动态查询,其实就是要实现拼接SQL语句。无论实现如何复杂,基本都是包括select的字段,from或者join的表,where或者having的条件。在Spring Data JPA有两种方法可以实现查询条件的动态查询,两种方法都用到了Criteria API。

Criteria API

这套API可用于构建对数据库的查询。

类型安全。通过定义元数据模型,在程序编译阶段就可以对类型进行检查,不像SQL需要与Mysql进行交互后才能发现类型问题。

如下即为元数据模型。创建一个元模型类,类名最后一个字符为下划线,内部的成员变量与UserInfo.class这个实体类的属性值相对应。

@StaticMetamodel(UserInfo.class)

public class UserInfo_ {

public static volatile SingularAttribute userId;

public static volatile SingularAttribute name;

public static volatile SingularAttribute age;

public static volatile SingularAttribute high;

}

可移植。API并不依赖具体的数据库,可以根据数据库类型的不同生成对应数据库类型的SQL,所以其为可移植的。

面向对象。Criteria API是使用的是各种类和对象如CriteriaQuery、Predicate等构建查询,是面向对象的。而如果直接书写SQL则相对于面向的是字符串。

第一种:通过JPA的Criteria API实现

EntityManager获取CriteriaBuilder

CriteriaBuilder创建CriteriaQuery

CriteriaQuery指定要查询的表,得到Root,Root代表要查询的表

CriteriaBuilder创建条件Predicate,Predicate相对于SQL的where条件,多个Predicate可以进行与、或操作。

通过EntityManager创建TypedQuery

TypedQuery执行查询,返回结果

public class UserInfoExtendDao {

@PersistenceContext(unitName = "springJpa")

EntityManager em;

public List getUserInfo(String name,int age,int high) {

CriteriaBuilder cb = em.getCriteriaBuilder();

CriteriaQuery query = cb.createQuery(UserInfo.class);

//from

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

//where

Predicate p1 = null;

if(name!=null) {

Predicate p2 = cb.equal(root.get(UserInfo_.name),name);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

if(age!=0) {

Predicate p2 = cb.equal(root.get(UserInfo_.age), age);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

if(high!=0) {

Predicate p2 = cb.equal(root.get(UserInfo_.high), high);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

query.where(p1);

List userInfos = em.createQuery(query).getResultList();

return userInfos;

}

}

第二种:DAO层接口实现JpaSpecificationExecutor接口

JpaSpecificationExecutor如下,方法参数Specification接口有一个方法toPredicate,返回值正好是Criteria API中的Predicate,而Predicate相对于SQL的where条件。与上一个方法相比,这种写法不需要指定查询的表是哪一张,也不需要自己通过Criteria API实现排序和分页,只需要通过新建Pageable、Sort对象并传参给findAll方法即可,简便一些。

public interface JpaSpecificationExecutor {

T findOne(Specification spec);

List findAll(Specification spec);

Page findAll(Specification spec, Pageable pageable);

List findAll(Specification spec, Sort sort);

long count(Specification spec);

}

UserInfoDao实现JpaSpecificationExecutor

public interface UserInfoDao

extends PagingAndSortingRepository, JpaSpecificationExecutor {}

实现Specification

public static Specification getSpec(final String name,final int age,final int high) {

return new Specification() {

@Override

public Predicate toPredicate(Root root, CriteriaQuery> query, CriteriaBuilder cb) {

Predicate p1 = null;

if(name!=null) {

Predicate p2 = cb.equal(root.get(UserInfo_.name),name);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

if(age!=0) {

Predicate p2 = cb.equal(root.get(UserInfo_.age), age);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

if(high!=0) {

Predicate p2 = cb.equal(root.get(UserInfo_.high), high);

if(p1 != null) {

p1 = cb.and(p1,p2);

} else {

p1 = p2;

}

}

return p1;

}

};

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值