Spring Data JPA实现动态查询的两种方法

作者:lulll 时间:2022-08-04 17:02:27 

前言

一般在写业务接口的过程中,很有可能需要实现可以动态组合各种查询条件的接口。如果我们根据一种查询条件组合一个方法的做法来写,那么将会有大量方法存在,繁琐,维护起来相当困难。想要实现动态查询,其实就是要实现拼接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<UserInfo, Integer> userId;
 public static volatile SingularAttribute<UserInfo, String> name;
 public static volatile SingularAttribute<UserInfo, Integer> age;
 public static volatile SingularAttribute<UserInfo, Long> high;
}

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

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

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

  1. EntityManager获取CriteriaBuilder

  2. CriteriaBuilder创建CriteriaQuery

  3. CriteriaQuery指定要查询的表,得到Root<UserInfo>,Root代表要查询的表

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

  5. 通过EntityManager创建TypedQuery

  6. TypedQuery执行查询,返回结果


public class UserInfoExtendDao {

@PersistenceContext(unitName = "springJpa")
EntityManager em;

public List<UserInfo> getUserInfo(String name,int age,int high) {
  CriteriaBuilder cb = em.getCriteriaBuilder();
  CriteriaQuery<UserInfo> query = cb.createQuery(UserInfo.class);

//from
  Root<UserInfo> 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<UserInfo> userInfos = em.createQuery(query).getResultList();
  return userInfos;
}
}

第二种:DAO层接口实现JpaSpecificationExecutor<T>接口

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


public interface JpaSpecificationExecutor<T> {
T findOne(Specification<T> spec);
List<T> findAll(Specification<T> spec);
Page<T> findAll(Specification<T> spec, Pageable pageable);
List<T> findAll(Specification<T> spec, Sort sort);
long count(Specification<T> spec);
}

UserInfoDao实现JpaSpecificationExecutor


public interface UserInfoDao
 extends PagingAndSortingRepository<UserInfo, String>, JpaSpecificationExecutor<UserInfo> {}

实现Specification


public static Specification<UserInfo> getSpec(final String name,final int age,final int high) {
  return new Specification<UserInfo>() {
    @Override
    public Predicate toPredicate(Root<UserInfo> 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;
    }
  };
}

项目代码:springdatajpademo_jb51.rar

来源:http://www.jianshu.com/p/45ad65690e33#

标签:spring,data,jpa
0
投稿

猜你喜欢

  • c# chart缩放,局部放大问题

    2021-09-26 20:46:40
  • C# 通过ServiceStack 操作Redis

    2023-12-13 06:18:01
  • 详解Java中Collections.sort排序

    2023-08-20 00:51:48
  • spring mvc中注解@ModelAttribute的妙用分享

    2022-09-15 13:53:25
  • 关于idea的gitignore文件编写及解决ignore文件不生效问题

    2023-02-28 02:04:13
  • Java如何优雅替换if-else语句

    2023-02-23 10:30:27
  • C# 7.0中解构功能详解

    2022-08-11 21:06:34
  • Java利用读写的方式实现音频播放代码实例

    2022-08-21 15:38:52
  • Java利用HttpClient模拟POST表单操作应用及注意事项

    2023-11-29 23:48:01
  • 详解Java的TCP/IP编程学习--基于定界符的成帧

    2023-11-23 03:02:32
  • java链式创建json对象的实现

    2023-11-12 12:36:51
  • Android实现价格走势自定义曲线图

    2023-04-03 23:15:06
  • C#实现关闭其他程序窗口或进程代码分享

    2022-05-26 07:38:03
  • SpringBoot整合Echarts实现用户人数和性别展示功能(详细步骤)

    2023-02-22 00:31:59
  • Java如何在沙箱环境中测试支付宝支付接口

    2023-11-02 14:55:15
  • java反射应用详细介绍

    2022-06-14 04:59:45
  • java实现分页显示效果

    2021-12-29 20:17:43
  • C# 常用日期时间函数(老用不熟)

    2021-08-21 10:12:18
  • java实现登录验证码功能

    2021-06-08 19:34:18
  • a2sd+状态下应用程序丢失的解决方法详细解析

    2023-04-20 16:08:11
  • asp之家 软件编程 m.aspxhome.com