SPring boot jpa 封装查询条件
最近使用spring data jpa做了两个项目,对于动态查询的不友好做了个类似hibernate的封装,记录也分享下
首先定义一个所有条件的容器,继承Specification
- /**
- * 定义一个查询条件容器
- * @author lee
- *
- * @param <T>
- */
- public class Criteria<T> implements Specification<T>{
- private List<Criterion> criterions = new ArrayList<Criterion>();
- public Predicate toPredicate(Root<T> root, CriteriaQuery<?> query,
- CriteriaBuilder builder) {
- if (!criterions.isEmpty()) {
- List<Predicate> predicates = new ArrayList<Predicate>();
- for(Criterion c : criterions){
- predicates.add(c.toPredicate(root, query,builder));
- }
- // 将所有条件用 and 联合起来
- if (predicates.size() > 0) {
- return builder.and(predicates.toArray(new Predicate[predicates.size()]));
- }
- }
- return builder.conjunction();
- }
- /**
- * 增加简单条件表达式
- * @Methods Name add
- * @Create In 2012-2-8 By lee
- * @param expression0 void
- */
- public void add(Criterion criterion){
- if(criterion!=null){
- criterions.add(criterion);
- }
- }
- }
然后是各种条件组装类,我首先做了一个接口来包装各种条件
- /**
- * 条件接口
- * 用户提供条件表达式接口
- * @Class Name Criterion
- * @Author lee
- * @Create In 2012-2-8
- */
- public interface Criterion {
- public enum Operator {
- EQ, NE, LIKE, GT, LT, GTE, LTE, AND, OR
- }
- public Predicate toPredicate(Root<?> root, CriteriaQuery<?> query,
- CriteriaBuilder builder);
- }
然后是针对不同类型条件处理的实现
一个是简单比较类型的处理
- /**
- * 简单条件表达式
- * @author lee
- *
- */
- public class SimpleExpression implements Criterion{
- private String fieldName; //属性名
- private Object value; //对应值
- private Operator operator; //计算符
- protected SimpleExpression(String fieldName, Object value, Operator operator) {
- this.fieldName = fieldName;
- this.value = value;
- this.operator = operator;
- }
- public String getFieldName() {
- return fieldName;
- }
- public Object getValue() {
- return value;
- }
- public Operator getOperator() {
- return operator;
- }
- @SuppressWarnings({ "rawtypes", "unchecked" })
- public Predicate toPredicate(Root<?> root, CriteriaQuery<?> query,
- CriteriaBuilder builder) {
- Path expression = null;
- if(fieldName.contains(".")){
- String[] names = StringUtils.split(fieldName, ".");
- expression = root.get(names[0]);
- for (int i = 1; i < names.length; i++) {
- expression = expression.get(names[i]);
- }
- }else{
- expression = root.get(fieldName);
- }
- switch (operator) {
- case EQ:
- return builder.equal(expression, value);
- case NE:
- return builder.notEqual(expression, value);
- case LIKE:
- return builder.like((Expression<String>) expression, "%" + value + "%");
- case LT:
- return builder.lessThan(expression, (Comparable) value);
- case GT:
- return builder.greaterThan(expression, (Comparable) value);
- case LTE:
- return builder.lessThanOrEqualTo(expression, (Comparable) value);
- case GTE:
- return builder.greaterThanOrEqualTo(expression, (Comparable) value);
- default:
- return null;
- }
- }
- }
一个逻辑条件计算实现
- /**
- * 逻辑条件表达式 用于复杂条件时使用,如但属性多对应值的OR查询等
- * @author lee
- *
- */
- public class LogicalExpression implements Criterion {
- private Criterion[] criterion; // 逻辑表达式中包含的表达式
- private Operator operator; //计算符
- public LogicalExpression(Criterion[] criterions, Operator operator) {
- this.criterion = criterions;
- this.operator = operator;
- }
- public Predicate toPredicate(Root<?> root, CriteriaQuery<?> query,
- CriteriaBuilder builder) {
- List<Predicate> predicates = new ArrayList<Predicate>();
- for(int i=0;i<this.criterion.length;i++){
- predicates.add(this.criterion[i].toPredicate(root, query, builder));
- }
- switch (operator) {
- case OR:
- return builder.or(predicates.toArray(new Predicate[predicates.size()]));
- default:
- return null;
- }
- }
- }
添加一个组装工厂类
- /**
- * 条件构造器
- * 用于创建条件表达式
- * @Class Name Restrictions
- * @Author lee
- */
- public class Restrictions {
- /**
- * 等于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression eq(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.EQ);
- }
- /**
- * 不等于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression ne(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.NE);
- }
- /**
- * 模糊匹配
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression like(String fieldName, String value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.LIKE);
- }
- /**
- *
- * @param fieldName
- * @param value
- * @param matchMode
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression like(String fieldName, String value,
- MatchMode matchMode, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return null;
- }
- /**
- * 大于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression gt(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.GT);
- }
- /**
- * 小于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression lt(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.LT);
- }
- /**
- * 大于等于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression lte(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.GTE);
- }
- /**
- * 小于等于
- * @param fieldName
- * @param value
- * @param ignoreNull
- * @return
- */
- public static SimpleExpression gte(String fieldName, Object value, boolean ignoreNull) {
- if(StringUtils.isEmpty(value))return null;
- return new SimpleExpression (fieldName, value, Operator.LTE);
- }
- /**
- * 并且
- * @param criterions
- * @return
- */
- public static LogicalExpression and(Criterion... criterions){
- return new LogicalExpression(criterions, Operator.AND);
- }
- /**
- * 或者
- * @param criterions
- * @return
- */
- public static LogicalExpression or(Criterion... criterions){
- return new LogicalExpression(criterions, Operator.OR);
- }
- /**
- * 包含于
- * @param fieldName
- * @param value
- * @return
- */
- @SuppressWarnings("rawtypes")
- public static LogicalExpression in(String fieldName, Collection value, boolean ignoreNull) {
- if(ignoreNull&&(value==null||value.isEmpty())){
- return null;
- }
- SimpleExpression[] ses = new SimpleExpression[value.size()];
- int i=0;
- for(Object obj : value){
- ses[i]=new SimpleExpression(fieldName,obj,Operator.EQ);
- i++;
- }
- return new LogicalExpression(ses,Operator.OR);
- }
- }
使用方法如下
- Criteria<Event> c = new Criteria<Event>();
- c.add(Restrictions.like("code", searchParam.getCode(), true));
- c.add(Restrictions.eq("level", searchParam.getLevel(), false));
- c.add(Restrictions.eq("mainStatus", searchParam.getMainStatus(), true));
- c.add(Restrictions.eq("flowStatus", searchParam.getFlowStatus(), true));
- c.add(Restrictions.eq("createUser.userName", searchParam.getCreateUser(), true));
- c.add(Restrictions.lte("submitTime", searchParam.getStartSubmitTime(), true));
- c.add(Restrictions.gte("submitTime", searchParam.getEndSubmitTime(), true));
- c.add(Restrictions.eq("needFollow", searchParam.getIsfollow(), true));
- c.add(Restrictions.ne("flowStatus", CaseConstants.CASE_STATUS_DRAFT, true));
- c.add(Restrictions.in("solveTeam.code",teamCodes, true));
- eventDao.findAll(c);
其中eventDao为继承JpaSpecificationExecutor的接口类
SPring boot jpa 封装查询条件的更多相关文章
- Spring Boot JPA的查询语句
文章目录 准备工作 Containing, Contains, IsContaining 和 Like StartsWith EndsWith 大小写不敏感 Not @Query Spring Boo ...
- spring boot jpa 复杂查询 动态查询 连接and和or 模糊查询 分页查询
最近项目中用到了jpa,刚接触的时候有些激动,以前的到层忽然不用写sql不用去自己实现了,只是取个方法名就实现了,太惊艳了,惊为天人,但是慢慢的就发现不是这么回事了,在动态查询的时候,不知道怎么操作了 ...
- 【Spring Data 系列学习】Spring Data JPA 自定义查询,分页,排序,条件查询
Spring Boot Jpa 默认提供 CURD 的方法等方法,在日常中往往时无法满足我们业务的要求,本章节通过自定义简单查询案例进行讲解. 快速上手 项目中的pom.xml.application ...
- Spring Boot 入门系列(二十七)使用Spring Data JPA 自定义查询如此简单,完全不需要写SQL!
前面讲了Spring Boot 整合Spring Boot JPA,实现JPA 的增.删.改.查的功能.JPA使用非常简单,只需继承JpaRepository ,无需任何数据访问层和sql语句即可实现 ...
- Spring NamedParameterJdbcTemplate命名参数查询条件封装, NamedParameterJdbcTemplate查询封装
Spring NamedParameterJdbcTemplate命名参数查询条件封装, NamedParameterJdbcTemplate查询封装 >>>>>> ...
- 序列化表单为json对象,datagrid带额外参提交一次查询 后台用Spring data JPA 实现带条件的分页查询 多表关联查询
查询窗口中可以设置很多查询条件 表单中输入的内容转为datagrid的load方法所需的查询条件向原请求地址再次提出新的查询,将结果显示在datagrid中 转换方法看代码注释 <td cols ...
- Spring Boot(五):Spring Boot Jpa 的使用
在上篇文章Spring Boot(二):Web 综合开发中简单介绍了一下 Spring Boot Jpa 的基础性使用,这篇文章将更加全面的介绍 Spring Boot Jpa 常见用法以及注意事项. ...
- Spring Boot Jpa 的使用
Spring Boot Jpa 介绍 首先了解 Jpa 是什么? Jpa (Java Persistence API) 是 Sun 官方提出的 Java 持久化规范.它为 Java 开发人员提供了一种 ...
- (转)Spring Boot(五):Spring Boot Jpa 的使用
http://www.ityouknow.com/springboot/2016/08/20/spring-boot-jpa.html 在上篇文章Spring Boot(二):Web 综合开发中简单介 ...
随机推荐
- Socket-window通讯
#define _WINSOCK_DEPRECATED_NO_WARNINGS #include<WINSOCK2.H> #include<STDIO.H> #include& ...
- Dubbo介绍(一)
Dubbo是一个分布式.高性能.透明化的 RPC 服务框架,作用是提供服务自动注册.自动发现等高效服务治理方案. 一.Dubbo架构图 Provider:提供者,服务发布方 Consumer:消费者, ...
- 八种排序算法原理及Java实现
原文链接:http://ju.outofmemory.cn/entry/372908
- vscode使用插件来添加文件说明和函数说明——42header——psioniq File Header——koroFileHeader
安装号以后,设置快捷键如下: 同时需要根据自己的需要的修改json文件 // 文件头部注释 "fileheader.customMade": { "Description ...
- 洛谷 U87561 魔法月饼
洛谷 U87561 魔法月饼 洛谷传送门 题目背景 \(9102\)年的中秋节注定与往年不同...因为在\(9102\)年的中秋节前夕,\(Seaway\)被告知今年的中秋节要新出一款月饼--魔法月饼 ...
- JDOJ 3055: Nearest Common Ancestors
JDOJ 3055: Nearest Common Ancestors JDOJ传送门 Description 给定N个节点的一棵树,有K次查询,每次查询a和b的最近公共祖先. 样例中的16和7的公共 ...
- 四则运算web版需求规格说明书
目录 1引言... 4 1.1 目的... 4 1.2 背景... 4 1.3 术语... 4 1.4 预期读者与阅读建议... 5 1.5 参考资料... 6 1.6 需求描述约定... ...
- C++面向对象程序设计学习笔记(4)
类与对象(2) string类 C++不仅向下兼容C的字符表示方法,也声明了一种更方便的字符串类型,即string类. 想要使用string类,必须包括头文件string,即要声明 #include& ...
- [Python] Python 模拟登录,并请求
Python 模拟登录,并请求 # encoding: utf- import requests import socket import time socket.setdefaulttimeout( ...
- [ Python入门教程 ] Python的控制语句
Python控制语句由条件语句.循环语句构成.控制语句根据条件表达式控制程序的流转.本章将介绍Python中控制语句的基本语法. 条件判断语句 (1)if条件语句 if语句用于检测某个条件是否成立.如 ...