spring data JPA 高级动态查询, 排序加分页加多参数查询 返回二元组

业务需求是根据前端传来的四个参数进行匹配查询, 并且要将总共查出来的数据总数返回至前端(因为前端有进行分页, 首尾页等), 但是前端未必会传来四个参数, 也可能是随机的其中一个,两个等. 这时候如果自己在dao层,手写查询方法或者sql语句需要大量代码, 并且非常麻烦, 在多方查阅资料,以及询问公司前辈的情况下, 终于完成了项目需求,

贴出前端查询页面:




贴出二元组代码:

package com.leadmap.mapservice.common;

public class TwoTuple<A, B> {

  public final A first;

    private final B second;

    public TwoTuple(A a, B b){

        first = a;

        second = b;

    }

    public A getFirst() {

        return first;

    }

    public B getSecond(){

        return second;

    }

    @Override

    public String toString(){

        return "(" + first + ", " + second + ")";

    }

}


贴出查询业务代码:

package com.leadmap.mapservice.service;

import com.leadmap.mapservice.common.TwoTuple;

import com.leadmap.mapservice.common.Util;

import com.leadmap.mapservice.dao.DocumentInfoDao;

import com.leadmap.mapservice.dao.OpinionFeedbackDao;

import com.leadmap.mapservice.entity.DocumentInfo;

import com.leadmap.mapservice.entity.OpinionFeedback;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.data.domain.PageRequest;

import org.springframework.data.domain.Pageable;

import org.springframework.data.jpa.domain.Specification;

import org.springframework.stereotype.Service;

import javax.persistence.criteria.CriteriaBuilder;

import javax.persistence.criteria.CriteriaQuery;

import javax.persistence.criteria.Predicate;

import javax.persistence.criteria.Root;

import java.util.ArrayList;

import java.util.Date;

import java.util.List;


@Service

public class DocumentService {

    @Autowired

    private DocumentInfoDao documentInfoDao;

    // 返回结果为实体类对象数组和匹配数据总数

    // 两个日期参数格式为 yyyy-MM-dd  数据库中对应字段格式为yyyy-mm-dd hh-mm-ss 因此下面会进行日期字符串拼接

    public TwoTuple<List<DocumentInfo>, Integer> getPageDocumentInfo(int start, int count, String beginTime, String endTime, String type, String title){

        // 创建分页参数

        Pageable page = new PageRequest(start, count);

        List<DocumentInfo> list = new ArrayList<>();

        // 新建查询参数, 重写方法

        Specification querySpecifi = new Specification<DocumentInfo>() {

            // 这几个参数算是固定用法吧 要用到就写上去

            @Override

            public Predicate toPredicate(Root root, CriteriaQuery criteriaQuery, CriteriaBuilder criteriaBuilder) {

                 // 姑且叫做 新建查询参数数组吧, 是多个查询条件组合的的数组

                List<Predicate> predicatesList = new ArrayList<>();

                // 当if成立 添加 匹配大于开始时间的查询参数

                if(!beginTime.isEmpty()){

                    Date beginDate = Util.StrToDate(beginTime+" 00:00:00");

                    predicatesList.add(criteriaBuilder.greaterThan(root.get("createTime"), beginDate));

                }

                // 同上 添加 匹配小于开始时间的查询参数  这两个参数组合也就相当于sql中的between...and...

                if(!endTime.isEmpty()){

                    Date endDate = Util.StrToDate(endTime + " 00:00:00");

                    predicatesList.add(criteriaBuilder.lessThan(root.get("createTime"), endDate));

                }

                // 添加匹配数据库中相同类型的参数

                if(!type.isEmpty()){

                    predicatesList.add(criteriaBuilder.like(root.get("type"),"%"+type+"%"));

                }

                if(!title.isEmpty()){

                    predicatesList.add(criteriaBuilder.like(root.get("title"), "%" + title + "%"));

                }

                // 设置排序条件 根据id倒序

                criteriaQuery.orderBy(criteriaBuilder.desc(root.get("id")));

                // 将排序与添加好的查询参数数组 作为返回值

                criteriaQuery.where(criteriaBuilder.and(predicatesList.toArray(new Predicate[predicatesList.size()])));

                return criteriaQuery.getRestriction();

            }

        };

        // 这个findAll是JpaSpecificationExecutor<T>接口的方法

        list = documentInfoDao.findAll(querySpecifi,page).getContent();

        // 拿到 匹配数据总数 此处以每页十条分页

        Integer counts = documentInfoDao.findAll(querySpecifi).size() / 10;

        // 返回二元组

        return new TwoTuple<List<DocumentInfo>, Integer>(list, counts);

    }

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,240评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,328评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,182评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,121评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,135评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,093评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,013评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,854评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,295评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,513评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,398评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,989评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,636评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,657评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352