JPA 通过Specification如何实现复杂查询


Posted in Java/Android onNovember 23, 2021

JPA 通过Specification实现复杂查询

JPA中继承BaseRepo之后,可以使用最基本的增删改查,如果想实现复杂查询,则需要借助Specification来完成这个功能:

下面就简单介绍一下Specification的使用

public void findAll(ConstructPlanPageReqEntity constructPlanPageReqEntity) {
	Integer pageNum = page.getPageNum();
        Integer pageSize = page.getPageSize();
        String costType = constructPlanPageReqEntity.getCostType();
        String name = constructPlanPageReqEntity.getName();
        String planMoneyStart = constructPlanPageReqEntity.getPlanMoneyStart();
        String planMoneyEnd = constructPlanPageReqEntity.getPlanMoneyEnd();
        String singMoneyEnd = constructPlanPageReqEntity.getSingMoneyEnd();
        String signMoneyStart = constructPlanPageReqEntity.getSignMoneyStart();
        long projectId = Long.parseLong(constructPlanPageReqEntity.getProjectId());
        String status = constructPlanPageReqEntity.getStatus();
        //分页
        pageNum=pageNum-1;
        Pageable pageable = PageRequest.of(pageNum, pageSize);
//多条件匹配查询
        Specification specification= new Specification<ContractPlanBean>() {
            @Override
            public Predicate toPredicate(Root<ContractPlanBean> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder criteriaBuilder) {
                ArrayList<Predicate> list = new ArrayList<>();
                Path<String> costType1 = root.get("costType");
                Path<String> name1 = root.get("name");
                Path<Long> projectId1 = root.get("projectId");
                Path<Object> status1 = root.get("status");
                if (projectId>0){
                    list.add(criteriaBuilder.equal(projectId1,projectId));
                }
                if (StringUtil.isNotEmpty(status)){
                    list.add(criteriaBuilder.equal(status1,status));
                }
                //条件查询
                if (StringUtil.isNotEmpty(costType)){
                    list.add(criteriaBuilder.equal(costType1,costType));
                }
                //模糊查询
                if (StringUtil.isNotEmpty(name)){
                    list.add(criteriaBuilder.like(name1,"%"+name+"%"));
                }
                //范围查询
                if (StringUtil.isNotEmpty(planMoneyStart)&&StringUtil.isNotEmpty(planMoneyEnd)){
                    try {
                        list.add(criteriaBuilder.between(root.get("planMoney"),NumberUtil.strToDouble(planMoneyStart),NumberUtil.strToDouble(planMoneyEnd)));
                    } catch (Exception e) {
                        throw new ApiException("规划金额查询失败");
                    }
                }
                 //排序
                criteriaQuery.orderBy(criteriaBuilder.asc(root.get("name")));
                Predicate[] array = new Predicate[list.size()];
                return criteriaBuilder.and(list.toArray(array));
            }
        };
}

以上代码实现了多条件查询,其中需要重写toPredicate方法,具体参数:

  • 用root.get()获取bean中的数据库对应字段
  • 用criteriaBuilder来组建条件查询语句

JPA 通过Specification如何实现复杂查询

上图是criteriaBuilder各种sql符号的方法名,根据需求组建不同的sql语句

criteriaBuilder.and(list.toArray(array))这句是最后定义各个sql查询条件的关系,这里用的and

至此,复杂sql语句就拼接完成,本人对Specification的使用未进行深入研究,个人觉得相对filter Strem的复杂查询来说Specification更繁琐,因此更倾向于通过Strem的复杂查询,这回就不多说了,下次就介绍下如何使用Stream进行复杂查询

spring-data-jpa Specification拼接复杂查询

public Page<ServiceItem> findAll(Map<String, String[]> params, ServiceItemConsumeStatus serviceItemConsumeStatus,ServiceItemStatus serviceItemStatus, Pageable pageable) {
        return dao.findAll(spec(serviceItemConsumeStatus, serviceItemStatus, params), pageable);
    }
    private Specification<ServiceItem> spec(final ServiceItemConsumeStatus serviceItemConsumeStatus, 
    final ServiceItemStatus serviceItemStatus, Map<String, String[]> params) {
        Collection<SearchFilter> filters = SearchFilter.parse(params).values();
        final Specification<ServiceItem> fsp = SearchFilter.spec(filters, ServiceItem.class);
        Specification<ServiceItem> sp = new Specification<ServiceItem>() {
            public Predicate toPredicate(Root<ServiceItem> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
                Predicate pred = fsp.toPredicate(root, query, cb);
                if (ServiceItemConsumeStatus.可消费.equals(serviceItemConsumeStatus)) {
                    pred = cb.and(pred, cb.gt(root.get("countLeft").as(int.class), 0));
                } else if (ServiceItemConsumeStatus.消费完毕.equals(serviceItemConsumeStatus)) {
                    pred = cb.and(pred, cb.le(root.get("countLeft").as(int.class), 0));
                }
                if (serviceItemStatus != null) {
                    pred = cb.and(pred, cb.equal(root.get("status"), serviceItemStatus));
                }
                return pred;
            }
        };
        return sp;
    }

以上为个人经验,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Java/Android 相关文章推荐
SpringBoot2 参数管理实践之入参出参与校验的方式
Jun 16 Java/Android
SpringBoot项目中控制台日志的保存配置操作
Jun 18 Java/Android
详解Java实现设计模式之责任链模式
Jun 23 Java/Android
spring boot项目application.properties文件存放及使用介绍
Jun 30 Java/Android
logback 实现给变量指定默认值
Aug 30 Java/Android
解决persistence.xml配置文件修改存放路径的问题
Feb 24 Java/Android
Java 写一个简单的图书管理系统
Apr 26 Java/Android
java版 简单三子棋游戏
May 04 Java/Android
Java 数组的使用
May 11 Java/Android
Java 轮询锁使用时遇到问题
May 11 Java/Android
SpringBoot项目多数据源及mybatis 驼峰失效的问题解决方法
Jul 07 Java/Android
java中如何截取字符串最后一位
Jul 07 Java/Android
Java使用JMeter进行高并发测试
Java 在线考试云平台的实现
OpenCV实现反阈值二值化
聊聊SpringBoot自动装配的魔力
Nov 17 #Java/Android
Springboot如何同时装配两个相同类型数据库
Nov 17 #Java/Android
OpenCV实现普通阈值
聊聊Lombok中的@Builder注解使用教程
Nov 17 #Java/Android
You might like
php实现文件下载更能介绍
2012/11/23 PHP
php将字符串转化成date存入数据库的两种方式
2014/04/28 PHP
php启用sphinx全文搜索的实现方法
2014/12/24 PHP
codeigniter中实现一次性加载多个view的方法
2015/03/20 PHP
YII Framework框架教程之日志用法详解
2016/03/14 PHP
无语,javascript居然支持中文(unicode)编程!
2007/04/12 Javascript
WEB 浏览器兼容 推荐收藏
2010/05/14 Javascript
IE6浏览器下resize事件被执行了多次解决方法
2012/12/11 Javascript
Javacript实现颜色梯度变化和渐变的效果代码
2013/05/31 Javascript
jQuery定义背景动态切换效果的方法
2015/03/23 Javascript
javascript控制层显示或隐藏的方法
2015/07/22 Javascript
JavaScript仿静态分页实现方法
2015/08/04 Javascript
详解javascript跨浏览器事件处理程序
2016/03/27 Javascript
详解Angular 4.x NgTemplateOutlet
2017/05/24 Javascript
使用webpack搭建react开发环境的方法
2018/05/15 Javascript
详解在React项目中安装并使用Less(用法总结)
2019/03/18 Javascript
React.js组件实现拖拽排序组件功能过程解析
2020/04/27 Javascript
JS+Canvas实现五子棋游戏
2020/08/26 Javascript
[01:01:43]EG vs VP 2018国际邀请赛淘汰赛BO3 第二场 8.24
2018/08/25 DOTA
Python EOL while scanning string literal问题解决方法
2020/09/18 Python
python中redis的安装和使用
2016/12/04 Python
Python实现对百度云的文件上传(实例讲解)
2017/10/21 Python
python使用PyCharm进行远程开发和调试
2017/11/02 Python
numpy.random.seed()的使用实例解析
2018/02/03 Python
python无限生成不重复(字母,数字,字符)组合的方法
2018/12/04 Python
pytorch判断是否cuda 判断变量类型方式
2020/06/23 Python
python写文件时覆盖原来的实例方法
2020/07/22 Python
html5中canvas学习笔记2-判断浏览器是否支持canvas
2013/01/06 HTML / CSS
德国自然时尚和有机产品购物网站:Waschbär
2019/05/29 全球购物
英国行业制服供应商:Alexandra
2019/09/14 全球购物
坚定理想信念心得体会
2014/03/11 职场文书
信息管理专业自荐书
2014/06/05 职场文书
2014机关党员干部“正风肃纪”思想汇报
2014/09/15 职场文书
2014年物业公司工作总结
2014/11/22 职场文书
教代会开幕词
2015/01/28 职场文书
党员反四风学习心得体会
2016/01/22 职场文书