Spring Data JPA最为优秀的特性就是能够经过自定义方法名称生成查询来轻松建立查询SQL。Spring Data JPA提供了一个Repository编程模型,最简单的方式就是经过扩展JpaRepository,咱们得到了一堆通用的CRUD方法,例如save,findAll,delete等。而且使用这些关键字能够构建不少的数据库单表查询接口:java
public interface CustomerRepository extends JpaRepository<Customer, Long> { Customer findByEmailAddress(String emailAddress); List<Customer> findByLastname(String lastname, Sort sort); Page<Customer> findByFirstname(String firstname, Pageable pageable); }
以上全部的查询都不用咱们手写SQL,查询生成器自动帮咱们工做,对于开发人员来讲只须要记住一些关键字,如:findBy、delete等等。可是,有时咱们须要建立复杂一点的查询,就没法利用查询生成器。可使用本节介绍的Specification来完成。spring
笔者仍是更愿意手写SQL来完成复杂查询,可是有的时候偶尔使用一下Specification来完成任务,也仍是深得我心。不排斥、不盲从。没有最好的方法,只有最合适的方法!数据库
是的,除了specification,咱们还可使用Criteria API构建复杂的查询,可是没有specification好用。咱们来看一下需求:在客户生日当天,咱们但愿向全部长期客户(2年以上)发送优惠券。咱们如何该检索Customer?编程
咱们有两个谓词查询条件:api
下面是使用JPA 2.0 Criteria API的实现方式:springboot
LocalDate today = new LocalDate(); CriteriaBuilder builder = em.getCriteriaBuilder(); CriteriaQuery<Customer> query = builder.createQuery(Customer.class); Root<Customer> root = query.from(Customer.class); Predicate hasBirthday = builder.equal(root.get(Customer_.birthday), today); Predicate isLongTermCustomer = builder.lessThan(root.get(Customer_.createdAt), today.minusYears(2); query.where(builder.and(hasBirthday, isLongTermCustomer)); em.createQuery(query.select(root)).getResultList();
LocalDate
用于比较客户的生日和今天的日期。em是javax.persistence.EntityManager此代码的主要问题在于,谓词查询条件不易于重用,您须要先设置 CriteriaBuilder, CriteriaQuery
,和Root
。另外,代码的可读性也不好。less
为了可以定义可重用谓词条件,咱们引入了Specification接口。学习
public interface Specification<T> { Predicate toPredicate(Root<T> root, CriteriaQuery query, CriteriaBuilder cb); }
结合Java 8的lambda表达式使用Specification接口时,代码变得很是简单ui
public CustomerSpecifications { //查询条件:生日为今天 public static Specification<Customer> customerHasBirthday() { return (root, query, cb) ->{ return cb.equal(root.get(Customer_.birthday), today); }; } //查询条件:客户建立日期在两年之前 public static Specification<Customer> isLongTermCustomer() { return (root, query, cb) ->{ return cb.lessThan(root.get(Customer_.createdAt), new LocalDate.minusYears(2)); }; } }
如今能够经过CustomerRepository执行如下操做:code
customerRepository.findAll(hasBirthday()); customerRepository.findAll(isLongTermCustomer());
咱们建立了能够单独执行的可重用谓词查询条件,咱们能够结合使用这些单独的谓词来知足咱们的业务需求。咱们可使用 and(…)
和 or(…)
链接specification。
customerRepository.findAll(where(customerHasBirthday()).and(isLongTermCustomer()));
与使用JPA Criteria API相比,它读起来很流利,提升了可读性并提供了更多的灵活性。