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 相关文章推荐
Java实战之用Swing实现通讯录管理系统
Jun 13 Java/Android
解析Java中的static关键字
Jun 14 Java/Android
Java 中的 Unsafe 魔法类的作用大全
Jun 26 Java/Android
SpringBoot实现异步事件驱动的方法
Jun 28 Java/Android
elasticSearch-api的具体操作步骤讲解
Jun 28 Java/Android
Java并发编程必备之Future机制
Jun 30 Java/Android
深入解读Java三大集合之map list set的用法
Nov 11 Java/Android
Java 超详细讲解IO操作字节流与字符流
Mar 25 Java/Android
Elasticsearch Recovery 详细介绍
Apr 19 Java/Android
Java 轮询锁使用时遇到问题
May 11 Java/Android
Java实现带图形界面的聊天程序
Jun 10 Java/Android
maven 解包依赖项中的文件的解决方法
Jul 15 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
第七节--类的静态成员
2006/11/16 PHP
php 向访客和爬虫显示不同的内容
2009/11/09 PHP
Zend Studio 实用快捷键一览表(精心整理)
2013/08/10 PHP
PHP判断JSON对象是否存在的方法(推荐)
2016/07/06 PHP
jQuery 技巧大全(新手入门篇)
2009/05/12 Javascript
jQuery 的全选(全非选)即取得被选中的值使用介绍
2013/11/12 Javascript
js使用正则实现ReplaceAll全部替换的方法
2014/08/22 Javascript
js实现同一页面多个运动效果的方法
2015/04/10 Javascript
如何实现移动端浏览器不显示 pc 端的广告
2015/10/15 Javascript
js实现html table 行,列锁定的简单实例
2016/10/13 Javascript
如何清除IE10+ input X 文本框的叉叉和密码输入框的眼睛图标
2016/12/21 Javascript
jQuery实现动态删除LI的方法
2017/05/30 jQuery
使用Angular CLI生成路由的方法
2018/03/24 Javascript
JS集合set类的实现与使用方法示例
2019/02/01 Javascript
了解重排与重绘
2019/05/29 Javascript
原生js实现3D轮播图
2020/03/21 Javascript
详解小程序如何改变onLoad的执行时机
2019/11/01 Javascript
只有 20 行的 JavaScript 模板引擎实例详解
2020/05/11 Javascript
[31:55]完美世界DOTA2联赛循环赛 IO vs GXR BO2第一场 11.04
2020/11/05 DOTA
[45:06]完美世界DOTA2联赛PWL S2 Magma vs InkIce 第二场 11.28
2020/12/02 DOTA
python寻找list中最大值、最小值并返回其所在位置的方法
2018/06/27 Python
Python将多个list合并为1个list的方法
2018/06/27 Python
Python实现的从右到左字符串替换方法示例
2018/07/06 Python
Python程序打包工具py2exe和PyInstaller详解
2019/06/28 Python
浅谈python3中input输入的使用
2019/08/02 Python
pytorch制作自己的LMDB数据操作示例
2019/12/18 Python
Python实现投影法分割图像示例(一)
2020/01/17 Python
Jupyter 无法下载文件夹如何实现曲线救国
2020/04/22 Python
纽约复古灵感的现代珠宝品牌:Lulu Frost
2018/03/03 全球购物
应届生求职简历的自我评价怎么写
2013/10/23 职场文书
4s店总经理岗位职责
2013/12/31 职场文书
实习单位接收函
2014/01/11 职场文书
新闻编辑专业自荐信
2014/07/02 职场文书
晚会闭幕词
2015/01/28 职场文书
2015公务员年度考核评语
2015/03/25 职场文书
请学会珍惜眼前,因为人生没有下辈子!
2019/11/12 职场文书