网站首页 > 开源技术 正文
Spring Data JPA最为优秀的特性就是可以通过自定义方法名称生成查询来轻松创建查询SQL。Spring Data JPA提供了一个Repository编程模型,最简单的方式就是通过扩展JpaRepository,我们获得了一堆通用的CRUD方法,例如save,findAll,delete等。并且使用这些关键字可以构建很多的数据库单表查询接口:
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);
}
- findByEmailAddress生成的SQL是根据email_address字段查询Customer表的数据
- findByLastname根据lastname字段查询Customer表的数据
- findByFirstname根据firstname字段查询Customer表的数据
以上所有的查询都不用我们手写SQL,查询生成器自动帮我们工作,对于开发人员来说只需要记住一些关键字,如:findBy、delete等等。但是,有时我们需要创建复杂一点的查询,就无法利用查询生成器。可以使用本节介绍的Specification来完成。
笔者还是更愿意手写SQL来完成复杂查询,但是有的时候偶尔使用一下Specification来完成任务,也还是深得我心。不排斥、不盲从。没有最好的方法,只有最合适的方法!
一、使用Criteria API构建复杂的查询
是的,除了specification,我们还可以使用Criteria API构建复杂的查询,但是没有specification好用。我们来看一下需求:在客户生日当天,我们希望向所有长期客户(2年以上)发送优惠券。我们如何该检索Customer?
我们有两个谓词查询条件:
- 生日
- 长期客户-2年以上的客户。
下面是使用JPA 2.0 Criteria API的实现方式:
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
- 下三行包含用于查询Customer实体的JPA基础结构实例的样板代码。
- 然后,在接下来的两行中,我们将构建谓词查询条件
- 在最后两行中,where用于连接两个谓词查询条件,最后一个用于执行查询。
此代码的主要问题在于,谓词查询条件不易于重用,您需要先设置 CriteriaBuilder, CriteriaQuery,和Root。另外,代码的可读性也很差。
二、specification
为了能够定义可重用谓词条件,我们可以使用Specification接口。
public interface Specification<T> {
Predicate toPredicate(Root<T> root, CriteriaQuery query, CriteriaBuilder cb);
}
结合Java 8的lambda表达式使用Specification接口时,代码变得非常简单
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执行以下操作:
customerRepository.findAll(hasBirthday());
customerRepository.findAll(isLongTermCustomer());
我们创建了可以单独执行的可重用谓词查询条件,我们可以结合使用这些单独的谓词来满足我们的业务需求。我们可以使用 and(…) 和 or(…)连接specification。
customerRepository.findAll(where(customerHasBirthday()).and(isLongTermCustomer()));
与使用JPA Criteria API相比,它读起来很流利,提高了可读性并提供了更多的灵活性。
- 上一篇: 系统应用根据Uri授予权限的正确姿势
- 下一篇: Aspose.Email V6.6.0发布
猜你喜欢
- 2024-12-14 ES的searchAfter使用详解
- 2024-12-14 Spring Boot Data Elasticsearch 通用工具类
- 2024-12-14 SpringBoot+Vue+ES 实现仿百度全文搜索
- 2024-12-14 Spring Data JPA——多表设计、一对多、多对多、多表查询
- 2024-12-14 一文读懂SpringBoot整合Elasticsearch
- 2024-12-14 ElasticSearchRepository和ElasticSearchTemplate的使用
- 2024-12-14 Elasticsearch 在地理信息空间索引的探索和演进
- 2024-12-14 Java微服务-一套前后台全部开源的H5商城送给大家(全部开源)
- 2024-12-14 android使用greendao来保存数据
- 2024-12-14 纯干货,Spring-data-jpa详解,全方位介绍
你 发表评论:
欢迎- 03-29蓝牙设备配对失败的系统性解决方案与技术解析
- 03-296MB PDF竟能运行Linux?这名高中生在PDF里玩DOOM后,再“整活”!
- 03-29平头哥玄铁RISC-V斩获MLPerf四项第一
- 03-29国产MCU先楫HPM6200系列发布:RISC-V架构,频率达600MHz
- 03-29switch怎么连接电视详细教程 switch oled快速连接电脑电视
- 03-29Switch系统更新至10.0.0版后:增加了控制器重新映射等功能
- 03-29Switch版《毁灭战士:永恒》过场动画仅20帧 开发商:画质高是这样的
- 03-29任天堂意外确认Switch 2的"C键"存在
- 最近发表
-
- 蓝牙设备配对失败的系统性解决方案与技术解析
- 6MB PDF竟能运行Linux?这名高中生在PDF里玩DOOM后,再“整活”!
- 平头哥玄铁RISC-V斩获MLPerf四项第一
- 国产MCU先楫HPM6200系列发布:RISC-V架构,频率达600MHz
- switch怎么连接电视详细教程 switch oled快速连接电脑电视
- Switch系统更新至10.0.0版后:增加了控制器重新映射等功能
- Switch版《毁灭战士:永恒》过场动画仅20帧 开发商:画质高是这样的
- 任天堂意外确认Switch 2的"C键"存在
- Switch NSP中文 毛茸茸的爪子Claws of Furry
- 三星展示可折叠手持设备,基本上是一个可以对折的Switch
- 标签列表
-
- jdk (81)
- putty (66)
- rufus (78)
- 内网穿透 (89)
- okhttp (70)
- powertoys (74)
- windowsterminal (81)
- netcat (65)
- ghostscript (65)
- veracrypt (65)
- asp.netcore (70)
- wrk (67)
- aspose.words (80)
- itk (80)
- ajaxfileupload.js (66)
- sqlhelper (67)
- express.js (67)
- phpmailer (67)
- xjar (70)
- redisclient (78)
- wakeonlan (66)
- tinygo (85)
- startbbs (72)
- webftp (82)
- vsvim (79)
本文暂时没有评论,来添加一个吧(●'◡'●)