SpringBoot+SpringDataJPA如何实现自定义查询[多表,多条件,分页,自定义sql封装]
举个例子:我们要在已经搭建好了的JPA环境下实现联合多表,多条件,多排序条件,分页查询一个表格数据,下面的表格

返回类MyJSON:
public class MyJSON {
private String code;
private String msg;
private Object data;
private Object extraData;
private Integer total;
public MyJSON(){
super();
}
public MyJSON(String code,String msg){
this.code=code;
this.msg=msg;
}
public MyJSON(String code, Object data) {
this.code = code;
this.data = data;
}
public MyJSON(String code, Object data, Integer total) {
this.code = code;
this.data = data;
this.total = total;
}
public MyJSON(String code, String msg, Object data) {
this.code = code;
this.msg = msg;
this.data = data;
}
public MyJSON(String code) {
this.code = code;
}
public MyJSON(String code, String msg, Object data, Integer total) {
this.code = code;
this.msg = msg;
this.data = data;
this.total = total;
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public Object getData() {
return data;
}
public void setData(Object data) {
this.data = data;
}
public Integer getTotal() {
return total;
}
public void setTotal(Integer total) {
this.total = total;
}
public Object getExtraData() {
return extraData;
}
public void setExtraData(Object extraData) {
this.extraData = extraData;
}
}
Code类:
public class Code {
public static final String SUCCESS="200";//查询结果成功
public static final String ERROR="201";//出错
public static final String SERVER_ERROR="202";//服务出错
public static final String SESSION_TIMEOUT="203";//session过期
}
MOrderModel:实体类
//lombok自行百度,其中的@Data大概作用就是省去get、set方法
import lombok.Data; @Data
public class MOrderModel {
private int orderId;//订单id
private Integer cusId;//顾客id
private String nickName;//顾客昵称
private String createTime;//下单时间
public MOrderModel() {
}
public MOrderModel(int orderId, Integer cusId, String nickName,String createTime) {
this.orderId = orderId;
this.cusId = cusId;
this.nickName = nickName;
this.createTime = createTime;
}
}
DataUtil:作用就是对从前端传过来需要排序的键值对的封装,返回字符串 “ order by AA desc,BB desc,CC asc ”,自己写order by也可以,接收map如:<"AA","desc">、<"BB","desc">、<"CC","asc">
import java.util.Map;
public class DataUtil {
public static String orderby(Map<String,String> sortOrderMap){
String queryOrder="";
if(sortOrderMap.size()!=0){
queryOrder+=" order by ";
}
for(Map.Entry<String, String> entry : sortOrderMap.entrySet()) {
queryOrder+=" "+entry.getKey()+" "+entry.getValue()+", ";
}
if(sortOrderMap.size()!=0){
queryOrder.substring(0,queryOrder.lastIndexOf(","));
}
return queryOrder;
}
}
Controller:
@PostMapping("findOrderList")
@ResponseBody
public MyJSON findOrderList(MOrderModel order, Integer pageIndex, Integer pageSize, String sortField, String sortOrder){
Map<String,String> sortMap=new HashMap<>();
if(StringUtils.isNotEmpty(sortField) && StringUtils.isNotEmpty(sortOrder)){
sortMap.put(sortField,sortOrder);
}
Page<MOrderModel> commodityPage=orderService.findDataList(order,pageIndex,pageSize,sortMap);
if(commodityPage!=null){
List<MOrderModel> list=commodityPage.getContent();
Object orderModel=list.get(0);
return new MyJSON(Code.SUCCESS,list,Integer.parseInt(commodityPage.getTotalElements()+""));
}else{
return new MyJSON(Code.ERROR);
}
}
ServiceImpl:
参数:
order:需要传过来的多条件查询
pageIndex:0开始的页码
pageSize:每页的条数
sortOrderMap是前端传过来的需要排序的表格列名及排序方向的键值对(多组合排序)
@Autowired
private MOrderRepository orderRepository;
public Page<MOrderModel> findDataList(MOrderModel order, Integer pageIndex, Integer pageSize,Map<String,String> sortOrderMap) {
Pageable pageable=PageRequest.of(pageIndex,pageSize);
StringBuffer querySql=new StringBuffer();
querySql.append("select * from order o " +
"where o.id=:id");//
querySql.append(DataUtil.orderby(sortOrderMap));//这里直接用封装好的order by
StringBuffer countSql=new StringBuffer();
countSql.append("select count(*) " +
"from order o " +
"where o.id=:id");
Map<String,Object> params=new HashMap<>();//封装参数
params.put("id",order.getId());
try {
Page<MOrderModel> page = orderRepository.findDataList(querySql.toString(), countSql.toString(), params, entityManager, pageable, MOrderModel.class);
return page;
}catch(Exception e){
e.printStackTrace();
return null;
}
}
MOrderRepository: 只需要继承一下自定义的BaseRepository就可以了,主要代码在BaseRepository中
import com.mercury.admin.entity.MOrder;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
import org.springframework.stereotype.Repository;
@Repository
public interface MOrderRepository<T> extends JpaSpecificationExecutor<MOrder>, JpaRepository<MOrder,Integer>,BaseRepository { }
BaseRepository :
import org.hibernate.query.internal.NativeQueryImpl;
import org.hibernate.transform.Transformers;
import org.springframework.cglib.beans.BeanMap;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.stereotype.Repository; import javax.persistence.EntityManager;
import javax.persistence.Query;
import java.math.BigInteger;
import java.sql.Timestamp;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Map;
@Repository
public interface BaseRepository { default <T> Page<T> findDataList(String querySql, String countSql, Map<String,Object> params, EntityManager entityManager, Pageable pageable,Class<T> clazz) throws IllegalAccessException, InstantiationException {
Query listQuery=entityManager.createNativeQuery(querySql);
listQuery.setFirstResult((int) pageable.getOffset());
listQuery.setMaxResults(pageable.getPageSize());
listQuery.unwrap(NativeQueryImpl.class).setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);//自动映射成map[稍微影响一点查询效率,但是封装后比较方便查看对应的字段,不需要再使用Object[] obj,obj[0],obj[1]...的方式]
for (Map.Entry<String, Object> entry : params.entrySet()) {//加入参数
listQuery.setParameter(entry.getKey(), entry.getValue());
}
Query countQuery=entityManager.createNativeQuery(countSql);
for (Map.Entry<String, Object> entry : params.entrySet()) {
countQuery.setParameter(entry.getKey(), entry.getValue());
}
List<Map<String,Object>> list= listQuery.getResultList();//由于上面已经将结果映射成了map所以这里直接转化成Map没问题
List<T> resultList=new ArrayList<>();
for(Map<String,Object> map:list){//遍历map将map转化为实体类bean
T bean=clazz.newInstance();//实例化T,可能会抛出两个异常IllegalAccessException、InstantiationException
for(Map.Entry<String,Object> entry:map.entrySet()){//格式化Timestamp为String类型,数据库中日期类型为Timestamp,在这里需要转化一下,直接在前端使用
if(entry.getValue() instanceof Timestamp){
try {
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date date=sdf.parse(entry.getValue()+"");
String dateStr = sdf.format(date);
map.put(entry.getKey(),dateStr);
} catch (ParseException e) {
e.printStackTrace();
} }
}
BeanMap.create(bean).putAll(map); resultList.add(bean);
}
BigInteger count=(BigInteger) countQuery.getSingleResult();
return new PageImpl<>(resultList, pageable, count.longValue());
}
}
SpringBoot+SpringDataJPA如何实现自定义查询[多表,多条件,分页,自定义sql封装]的更多相关文章
- sqlserver查询所有表的行数的sql语句
原文:sqlserver查询所有表的行数的sql语句 select object_name(id),rowcnt from sysindexes where indid<2 and object ...
- 从数据库中查询所有表及所有字段的SQL语句
从数据库中查询所有表及所有字段的SQL语句 由于一个小项目的需要,近日完成一个从数据库中查询所有表及所有字段的方法,其实用两条SQL语句就可以完成. Sql Server版:列出当前DB中所有表:se ...
- Sharding-Jdbc 自定义分库分表-复合分片算法自定义实现
Sharding-JDBC中的分片策略有两个维度,分别是: 数据源分片策略(DatabaseShardingStrategy) 表分片策略(TableShardingStrategy) 其中,数据源分 ...
- 几种常见数据库查询判断表和字段是否存在sql
1.MSSQL Server 表:select COUNT(*) from dbo.sysobjectsWHEREname= 'table_name': 字段:select COUNT(*) ...
- C# MongoDB 查询,分组,聚合,排序,条件,分页
先下载个C#的驱动.MongoDB提供各种主流与非主流预言的开发驱动. C# Driver 下载地址:这里 CSharp Driver Tutorial:这里 下载文件安装或者解压缩包 如果您是安装, ...
- 自定义MVC框架之工具类-分页类的封装
以前写过一个MVC框架,封装的有点low,经过一段时间的沉淀,打算重新改造下,之前这篇文章封装过一个验证码类. 这次重新改造MVC有几个很大的收获 >全部代码都是用Ubuntu+Vim编写,以前 ...
- MySQL查询数据表中数据记录(包括多表查询)
MySQL查询数据表中数据记录(包括多表查询) 在MySQL中创建数据库的目的是为了使用其中的数据. 使用select查询语句可以从数据库中把数据查询出来. select语句的语法格式如下: sele ...
- 数据库表设计时一对一关系存在的必要性 数据库一对一、一对多、多对多设计 面试逻辑题3.31 sql server 查询某个表被哪些存储过程调用 DataTable根据字段去重 .Net Core Cors中间件解析 分析MySQL中哪些情况下数据库索引会失效
数据库表设计时一对一关系存在的必要性 2017年07月24日 10:01:07 阅读数:694 在表设计过程中,我无意中觉得一对一关系觉得好没道理,直接放到一张表中不就可以了吗?真是说,网上信息什么都 ...
- sql中对查询出来的数据进行分页
当sql中存储的数据量比较大时,在web中 数据显示时都会对数据进行分页,分页不会在客户端进行分页,而是在数据库查询过程中进行了分页. sql代码: DECLARE @pageindex INT; - ...
随机推荐
- SpringBoot学习(一)—— idea 快速搭建 Spring boot 框架
简介 优点 Spring Boot 可以以jar包的形式独立运行,运行一个Spring Boot 项目只需要通过 java -jar xx.jar 来运行. Spring Boot 可以选择内嵌Tom ...
- python+appium搭建的测试环境
: 1,安装jdk JDK下载好jdk直接点下一步就可以了,然后开始配置变量classpath, path, Java_home:再运行cmd,并输入Java和javac看输出判断环境变量是否配好了. ...
- python logger日志通用配置文件
阅读须知⚠️ 1.示例代码可直接放在项目py文件中即可使用 2.project_name,logfile_name变量需根据你的项目进行修改 3.日志输出格式format选择(可根据你的需要替换或修改 ...
- 【Luogu P1878】舞蹈课
Luogu P1878 事实上这道题并不难,但我真没弄懂我手写堆为什么过不了.所以 STL大法好!!! 基本思路 对于每一对相邻异性,将他们的舞蹈技术的差插入一个堆 通过维护这个小根堆,每次就可以取得 ...
- JavaScript笔记五
1.条件分支语句 - switch语句 - 语法: switch(条件表达式){ case 表达式: 语句... break; case 表达式: 语句... break; case 表达式: 语句. ...
- c#关于数据和方法在不同类中的引用-xdd
关于数据和方法在不同类中的引用 using System; using System.Collections.Generic; using System.Linq; using System.Text ...
- linux计算机网络基础
OSI7层协议和TCP/IP4层网络协议 第一层:物理层,定义各种物理设备的规范,如通信距离,接口大小等. 第二层:数据链路层,基于mac地址通信是,数据报文封装和相应方式. 第三层:网络层,基于IP ...
- CSS中如果实现元素浮动,看这篇文章就足够了
浮动基本介绍 在标准文档流中元素分为2种,块级元素和行内元素,如果想让一些元素既要有块级元素的特点也同时保留行内元素特点,只能让这些元素脱离标准文档流即可. 浮动可以让元素脱离标准文档流,可以实现让多 ...
- 记一个bootloader的cache问题
问题背景 最近往一个armv7板子的bootloader中移植了解压算法,移植本身还比较顺利,但移植完了发现,功能是正常的,但效率大打折扣.解压同样的数据,耗时大约是uboot的10倍. 初步定位 从 ...
- C语言之修改常量
前言:指针!菜鸟的终点,高手的起点.漫谈一些进阶之路上的趣事:记录一些语言本身的特性以及思想,没有STL,也没有API! 0x01: 程序内存中的存储划分 对于程序在内存中是如何分布的,网上有多个解释 ...