如何使用 JPA 实现分页查询并返回 VO 对象

news2025/1/5 8:14:42

JPA分页踩坑指南

1.原生sql查询返回vo类包含主键id,无法自动映射,需要用到投影ResultTransformer,所以我定义了一个投影工具类

JpaCommonService

2.异步调用原生查询方法的时候,需要用

NativeQuery<?> query = entityManager.createNativeQuery(sql).unwrap(NativeQuery.class);,不能用NativeQueryImpl<?> query = entityManager.createNativeQuery(sql).unwrap(NativeQueryImpl.class);

3.用jpql查询entityManager.createQuery()分页查询,如分页会涉及到子查询是不支持的,只能用原生sql, 不支持select count(1) from (select * from User) t


以下是分页返回vo的具体实现

分页返回vo

下面是用到的一些类和方法

分页参数
package com.example.springbootjpadruid.domain.common;

import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class BasePage {

    private int page = 1;

    private int size = 10;

    /**
     * 是否需要分页
     */
    private boolean needPage = false;

    public int getPage() {
        if (page < 1) {
            return 0;
        } else {
            return page - 1;
        }
    }

    public int getOffset() {
        return getPage() * size;
    }
}

分页返回结果
package com.example.springbootjpadruid.domain.common;

import cn.hutool.core.bean.BeanUtil;
import lombok.Data;
import org.springframework.data.domain.Page;

import java.util.List;

@Data
public class PageVO<T> {

    private int page = 1;

    private int size = 10;

    private long total;

    private int totalPage;

    private List<T> list;


    public static <S, T> PageVO<T> of(Page<S> page, Class<T> clazz) {
        PageVO<T> pageVO = new PageVO<>();
        pageVO.setPage(page.getNumber() + 1);
        pageVO.setSize(page.getSize());
        pageVO.setTotal(page.getTotalElements());
        List<S> content = page.getContent();
        pageVO.setList(BeanUtil.copyToList(content, clazz));
        pageVO.setTotalPage(page.getTotalPages());
        return pageVO;
    }

    public static <T> PageVO<T> of(Page<T> page) {
        PageVO<T> pageVO = new PageVO<>();
        pageVO.setPage(page.getNumber() + 1);
        pageVO.setSize(page.getSize());
        pageVO.setTotal(page.getTotalElements());
        pageVO.setList(page.getContent());
        pageVO.setTotalPage(page.getTotalPages());
        return pageVO;
    }
}

原生sql查询通用方法
package com.example.springbootjpadruid.service;

import com.alibaba.fastjson.JSON;
import com.example.springbootjpadruid.domain.common.BasePage;
import org.hibernate.query.NativeQuery;
import org.hibernate.transform.ResultTransformer;
import org.springframework.beans.SimpleTypeConverter;
import org.springframework.stereotype.Service;

import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.persistence.Query;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

/**
 * @author Jonny
 * @description 原生sql查询,自动将下划线转驼峰
 */
@Service
public class JpaCommonService {

    private static final SimpleTypeConverter CONVERTER = new SimpleTypeConverter();

    @PersistenceContext
    private EntityManager entityManager;

    /**
     * 查询数据集合
     *
     * @param sql    查询sql,参数用:name格式
     * @param params 查询参数map格式,key对应参数中的:name
     * @param clazz  实体类型,为空则直接转换为map格式
     * @return list
     */
    @SuppressWarnings("unchecked")
    public <P extends BasePage> List<?> queryList(String sql, P params, Class<?> clazz) {
        String jsonParam = JSON.toJSONString(params);
        Map<String, Object> mapParams = JSON.parseObject(jsonParam, Map.class);
        // 提取 SQL 中的参数并过滤无效参数
        Set<String> sqlParams = extractSqlParams(sql);
        Map<String, Object> filteredParams = filterParams(sqlParams, mapParams);
        return queryList(sql, mapParams, filteredParams, clazz);
    }

    /**
     * 总记录数
     *
     * @param sql    查询sql,参数用:name格式
     * @param params 查询参数map格式,key对应参数中的:name
     * @return list
     */
    @SuppressWarnings("unchecked")
    public <P extends BasePage> long count(String sql, P params) {
        String jsonParam = JSON.toJSONString(params);
        Map<String, Object> mapParams = JSON.parseObject(jsonParam, Map.class);
        // 提取 SQL 中的参数并过滤无效参数
        Set<String> sqlParams = extractSqlParams(sql);
        Map<String, Object> filteredParams = filterParams(sqlParams, mapParams);
        return count(sql, mapParams, filteredParams, Long.class);
    }

    /**
     * 查询数据集合
     *
     * @param sql            查询sql,参数用:name格式
     * @param params         查询参数map格式,key对应参数中的:name
     * @param filteredParams sql中使用的参数
     * @param clazz          实体类型,为空则直接转换为map格式
     * @return list
     */
    @SuppressWarnings("all")
    public List<?> queryList(String sql, Map<String, Object> params, Map<String, Object> extractSqlParams, Class<?> clazz) {
        try {
            NativeQuery<?> query = entityManager.createNativeQuery(sql).unwrap(NativeQuery.class);
            if (Objects.nonNull(extractSqlParams)) {
                extractSqlParams.forEach((k, v) -> query.setParameter(k, v));
            }
            needPage(params, query);
            if (Objects.isNull(clazz)) {
                query.unwrap(NativeQuery.class).setResultTransformer(new AliasToEntityMapResultTransformer());
                return query.getResultList();
            } else {
                query.unwrap(NativeQuery.class).setResultTransformer(new AliasToBeanResultTransformer(clazz));
                return query.getResultList();
            }
        } catch (Exception e) {
            throw new RuntimeException("查询或转换数据时出错", e);
        }
    }

    @SuppressWarnings("all")
    public long count(String countSql, Map<String, Object> params, Map<String, Object> filteredParams, Class<?> clazz) {
        long total;
        try {
            Query countQuery = entityManager.createNativeQuery(countSql);
            if (Objects.nonNull(filteredParams)) {
                filteredParams.forEach(countQuery::setParameter);
            }
            total = Long.parseLong(countQuery.getSingleResult().toString());
        } catch (Exception e) {
            throw new RuntimeException("查询记录数出错", e);
        }
        return total;
    }

    /**
     * 判断是否需要分页
     * 当需要分页时,需要设置setNeedPage(true),并设置offset和size
     *
     * @param params 查询参数
     * @param query  查询对象
     */
    private void needPage(Map<String, Object> params, Query query) {
        if (Boolean.TRUE.equals(params.get("needPage"))) {
            int offset = (int) params.get("offset");
            int size = (int) params.get("size");
            query.setFirstResult(offset);
            query.setMaxResults(size);
        }
    }

    /**
     * 从 SQL 中提取所有命名参数
     *
     * @param sql SQL 查询
     * @return 参数名集合
     */
    private static Set<String> extractSqlParams(String sql) {
        Pattern pattern = Pattern.compile(":(\\w+)");
        Matcher matcher = pattern.matcher(sql);

        Set<String> params = new HashSet<>();
        while (matcher.find()) {
            params.add(matcher.group(1));
        }
        return params;
    }

    /**
     * 根据 SQL 中的参数过滤多余的 Map 参数
     *
     * @param sqlParams SQL 中使用的参数
     * @param params    原始参数
     * @return 过滤后的参数
     */
    private static Map<String, Object> filterParams(Set<String> sqlParams, Map<String, Object> params) {
        if (params == null || params.isEmpty()) {
            return Collections.emptyMap();
        }
        return params.entrySet().stream().filter(entry -> sqlParams.contains(entry.getKey())).collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
    }

    private static class AliasToEntityMapResultTransformer implements ResultTransformer {

        @Override
        public Object transformTuple(Object[] tuple, String[] aliases) {
            Map<String, Object> result = new HashMap<>();
            for (int i = 0; i < aliases.length; i++) {
                if (aliases[i] != null) {
                    result.put(aliases[i].toLowerCase(), tuple[i]);
                }
            }
            return result;
        }

        @Override
        public List transformList(List collection) {
            return collection;
        }
    }

    private static class AliasToBeanResultTransformer implements ResultTransformer {

        private final Class<?> resultClass;

        public AliasToBeanResultTransformer(Class<?> resultClass) {
            this.resultClass = resultClass;
        }

        /**
         * 转换结果集 自定将下划线转驼峰
         *
         * @param tuple   结果集
         * @param aliases 字段名
         * @return 实体对象
         */
        @Override
        public Object transformTuple(Object[] tuple, String[] aliases) {
            try {
                Object result = resultClass.getDeclaredConstructor().newInstance();
                PropertyDescriptor[] props = Introspector.getBeanInfo(resultClass).getPropertyDescriptors();
                Map<String, Method> writeMethodMap = Arrays.stream(props).filter(p -> Objects.nonNull(p.getWriteMethod())).collect(Collectors.toMap(p -> p.getName().toLowerCase(), PropertyDescriptor::getWriteMethod));

                for (int i = 0; i < aliases.length; i++) {
                    if (aliases[i] == null) {
                        continue;
                    }
                    String fieldName = aliases[i].toLowerCase().replace("_", "");
                    Method writeMethod = writeMethodMap.get(fieldName);
                    if (writeMethod != null) {
                        Object value = CONVERTER.convertIfNecessary(tuple[i], writeMethod.getParameterTypes()[0]);
                        writeMethod.invoke(result, value);
                    }
                }
                return result;
            } catch (Exception e) {
                throw new RuntimeException("实体映射错误: " + resultClass.getName(), e);
            }
        }

        @Override
        public List transformList(List collection) {
            return collection;
        }
    }
}

使用方法
package com.example.springbootjpadruid.service.impl;

import com.example.springbootjpadruid.domain.common.PageVO;
import com.example.springbootjpadruid.domain.entity.primary.User;
import com.example.springbootjpadruid.domain.query.UserQuery;
import com.example.springbootjpadruid.domain.vo.UserVO;
import com.example.springbootjpadruid.repository.primary.UserRepository;
import com.example.springbootjpadruid.service.JpaCommonService;
import com.example.springbootjpadruid.service.UserService;
import com.github.wenhao.jpa.PredicateBuilder;
import com.github.wenhao.jpa.Specifications;
import org.hibernate.query.NativeQuery;
import org.hibernate.transform.ResultTransformer;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.stereotype.Service;
import org.springframework.util.StringUtils;

import javax.annotation.Resource;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.persistence.Query;
import java.util.List;

@Service
public class UserServiceImpl implements UserService {

    @Resource
    private UserRepository repository;

    @PersistenceContext
    private EntityManager entityManager;

    @Resource
    private JpaCommonService jpaCommonService;

    @Override
    public String init(List<User> users) {
        repository.saveAll(users);
        return "ok";
    }

    @Override
    public User getUser() {
        PredicateBuilder<User> where = Specifications.and();
        where.eq("username", "zhangsan");
        where.eq("age", 20);
        return repository.findOne(where.build()).orElse(null);
    }

    /**
     * 分页1
     *
     * @param param
     * @return
     */
    @Override
    public PageVO<UserVO> listPage(UserQuery param) {
        PageRequest pageable = PageRequest.of(param.getPage(), param.getSize());
        Page<User> page = repository.findAll(pageable);
        return PageVO.of(page, UserVO.class);
    }

    /**
     * 分页2
     *
     * @param param 查询参数
     * @return 分页数据
     */
    @SuppressWarnings("all")
    @Override
    public PageVO<UserVO> nativeListPage(UserQuery param) {
        StringBuffer sql = new StringBuffer("select id as id, username as username, age as age, address as address from sys_user where 1 = 1");
        buildWhere(param, sql);

        StringBuffer countSql = new StringBuffer("select count(1) from sys_user where 1 = 1");
        buildWhere(param, countSql);

        PageRequest pageable = PageRequest.of(param.getPage(), param.getSize());

        NativeQuery query = entityManager.createNativeQuery(sql.toString()).unwrap(NativeQuery.class);
        query.setFirstResult(param.getOffset());
        query.setMaxResults(param.getSize());
        // 主键id无法直接映射,需要手动设置
        // query.setResultTransformer(Transformers.aliasToBean(UserVO.class));

        // 由于返回数据包含主键id,无法直接映射,只能借助投影手动映射
        query.setResultTransformer(new ResultTransformer() {
            @Override
            public Object transformTuple(Object[] objects, String[] strings) {
                UserVO userVO = new UserVO();
                userVO.setId(Long.parseLong(objects[0].toString()));
                userVO.setUsername(objects[1].toString());
                userVO.setAge(Integer.parseInt(objects[2].toString()));
                userVO.setAddress(objects[3].toString());
                return userVO;
            }

            @Override
            public List transformList(List list) {
                return list;
            }
        });
        buildParam(param, query);

        Query countQuery = entityManager.createNativeQuery(countSql.toString());
        buildParam(param, countQuery);
        long total = Long.parseLong(countQuery.getSingleResult().toString());

        PageImpl<UserVO> pageImpl = new PageImpl<>(query.getResultList(), pageable, total);
        return PageVO.of(pageImpl);
    }

    /**
     * 分页3
     * 原生sql自定转驼峰,带下划线不管大小自动转标准驼峰,全大写会转小写然后转标准驼峰
     * 解决分页方法2的原生sql如果没用as别名,无法自动转驼峰的问题
     *
     * @param param
     * @return
     */
    @SuppressWarnings("all")
    @Override
    public PageVO<UserVO> listPageByJpaUtil(UserQuery param) {
        // StringBuffer sql = new StringBuffer("select id, username, age, address, user_role as USER_ROLE from sys_user where 1 = 1");
        StringBuffer sql = new StringBuffer("select * from sys_user where 1 = 1");
        buildWhere(param, sql);

        StringBuffer countSql = new StringBuffer("select count(1) from sys_user where 1 = 1");
        buildWhere(param, countSql);

        PageRequest pageable = PageRequest.of(param.getPage(), param.getSize());
        param.setNeedPage(Boolean.TRUE);
        List<UserVO> userList = (List<UserVO>) jpaCommonService.queryList(sql.toString(), param, UserVO.class);
        long total = jpaCommonService.count(countSql.toString(), param);

        PageImpl<UserVO> pageImpl = new PageImpl<>(userList, pageable, total);
        return PageVO.of(pageImpl);
    }

    private void buildParam(UserQuery param, Query query) {
        if (StringUtils.hasText(param.getUsername())) {
            query.setParameter("username", param.getUsername());
        }
    }

    private void buildWhere(UserQuery param, StringBuffer sql) {
        if (StringUtils.hasText(param.getUsername())) {
            sql.append(" and username = :username");
        }
    }

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2269840.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

权限菜单之菜单管理 SpringBoot + VUE

一、 数据表设计 新建表sys_menu 表内数据 添加实体类Menu package com.example.demo.demos.web.demo.entity;import com.baomidou.mybatisplus.annotation.IdType; import com.baomidou.mybatisplus.annotation.TableField; import com.baomidou.mybatisplus.annotation.Ta…

【AIGC篇】AIGC 引擎:点燃创作自动化的未来之火

&#xff1a;羑悻的小杀马特.-CSDN博客 未来都是惊喜。你生来本应为高山。并非草芥。 引言&#xff1a; 在当今数字化的时代&#xff0c;人工智能生成内容&#xff08;AIGC&#xff09;正以一种前所未有的力量改变着我们的创作领域。它就像一个神秘而强大的魔法师&#xff0c;…

UnityRenderStreaming使用记录(三)

测试UnityRenderStreaming在Ubuntu24.04.1LTS上的表现 先放上运行图操作系统 Ubuntu24.04.1LTSUnity测试工程环境相关修改遇到的问题 先放上运行图 操作系统 Ubuntu24.04.1LTS 系统下载地址 https://cn.ubuntu.com/download/desktop安装UnityHub https://blog.csdn.net/AWNUXC…

从0开始的docker镜像制作-ubuntu22.04

从0开始的docker镜像制作-ubuntu22.04 一、拉取基础ubuntu22.04镜像二、进入拉取的docker镜像中&#xff0c;下载自己需要的安装包三、安装需要的系统软件四、打包现有镜像为一个新的镜像五、推送打包的镜像到私有docker服务器1.编辑docker文件&#xff0c;使其允许http传输和对…

多模态论文笔记——CogVLM和CogVLM2(副)

大家好&#xff0c;这里是好评笔记&#xff0c;公主号&#xff1a;Goodnote&#xff0c;专栏文章私信限时Free。本文详细介绍多模态模型的LoRA版本——CogVLM和CogVLM2。在SD 3中使用其作为captioner基准模型的原因和优势。 文章目录 CogVLM论文背景VLMs 的任务与挑战现有方法及…

gitlab-runner的卸载与安装

如果你使用rpm方式安装gitlab-runner&#xff0c;则可以参考本教程。 卸载 停止和卸载gitlab-runner 停止 gitlab-runner stopchkconfig gitlab-runner off卸载 gitlab-runner uninstall删除rpm包 查询出rpm包名&#xff0c;根据包名删除rpm。 [rootEuler02 ~]# rpm -qa …

Nacos配置中心总结

Nacos配置中心总结 Nacos配置文件的加载顺序和优先级 加载顺序 nacos作为配置中心时&#xff0c;需要在bootstrap.yml文件中添加nacos config相关的配置&#xff0c;这样系统启动时就能先去拉取nacos server上的配置了。拉取过来后会和本地配置文件进行合并。 bootstrap.ym…

赛博周刊·2024年度工具精选(图片资源类)

1、EmojiSpark emoji表情包查找工具。 2、fluentui-emoji 微软开源的Fluent Emoji表情包。 3、开源Emoji库 一个开源的emoji库&#xff0c;目前拥有4000个emoji表情。 4、中国表情包大合集博物馆 一个专门收集中国表情包的项目&#xff0c;已收录5712张表情包&#xff0c;并…

Goland:专为Go语言设计的高效IDE

本文还有配套的精品资源&#xff0c;点击获取 简介&#xff1a;Goland是JetBrains公司开发的集成开发环境&#xff08;IDE&#xff09;&#xff0c;专为Go语言设计&#xff0c;提供了高效的代码编辑、强大的调试工具和丰富的项目管理功能。其智能代码补全、强大的调试与测试支…

小程序发版后,用户使用时,强制更新为最新版本

为什么要强制更新为最新版本&#xff1f; 在小程序的开发和运营过程中&#xff0c;强制用户更新到最新版本是一项重要的策略&#xff0c;能够有效提升用户体验并保障系统的稳定性与安全性。以下是一些主要原因&#xff1a; 1. 功能兼容 新功能或服务通常需要最新版本的支持&…

EasyExcel(环境搭建以及常用写入操作)

文章目录 EasyExcel环境搭建1.创建模块 easyexcel-demo2.引入依赖3.启动类创建 EasyExcel写1.最简单的写入1.模板2.方法3.结果 Write01.xlsx 2.指定字段不写入Excel1.模板2.方法3.结果 Write02.xlsx 3.指定字段写入excel1.模板2.方法3.结果 Write03.xlsx 4.按照index顺序写入ex…

典型常见的基于知识蒸馏的目标检测方法总结三

来源&#xff1a;Google学术2023-2024的顶会顶刊论文 NeurIPS 2022&#xff1a;Towards Efficient 3D Object Detection with Knowledge Distillation 为3D目标检测提出了一种知识蒸馏的Benchmark范式&#xff0c;包含feature的KD&#xff0c;Logit的cls和reg的KD&#xff0c…

通过Dockerfile来实现项目可以指定读取不同环境的yml包

通过Dockerfile来实现项目可以指定读取不同环境的yml包 1. 挂载目录2. DockerFile3. 运行脚本deploy.sh4. 运行查看日志进入容器 5. 接口测试修改application-dev.yml 6. 优化Dockerfile7. 部分参数解释8. 优化不同环境下的日志也不同调整 Dockerfile修改部署脚本 deploy.sh重新…

开源的go语言统一配置中心 - nacos + nacos go sdk

配置文件实时更新机制的场景需求 配置文件热更新主要应用于需要在不停机的情况下动态调整系统行为的场景&#xff0c;例如修改服务参数、切换数据源等。其原理在于通过一个中心化的管理平台来存储和分发最新的配置信息。当配置文件发生变化时&#xff0c;该平台会主动或被动地…

对45家“AI+安全”产品/方案的分析

一. 关键洞察 “AI+安全”创新非常活跃,一片百家争鸣之势,赛道选择上,以事件分诊Incident Triage、 安全辅助Security Copilots、自动化Automation三者为主为主,这充分反映了当前安全运营的主要需求,在产品理念选择上以 AI 和 自动化为主,这确实又切合上了在关键…

GESP202412 三级【数字替换】题解(AC)

》》》点我查看「视频」详解》》》 [GESP202412 三级] 数字替换 题目描述 小杨有一个包含 n n n 个数字的序列 A A A&#xff0c;即 A [ a 1 , a 2 , … , a n ] A[a_1,a_2,\ldots,a_n] A[a1​,a2​,…,an​]&#xff0c;他想将其中大于 k k k 的数字都替换为序列的最大…

springboot集成websokcet+H5开发聊天原型(二)

本文没有写完~~~~ 聊天相关数据结构&#xff1a; 我们初步设计了如下几个数据结构。 //存放 sessionId 与 userId 的map private Map<String,String> sessionId_userId new HashMap<>(); // 用于存储用户与群组的关联关系&#xff0c;键为用户ID&#xff0c;值…

List接口(源码阅读)

文章目录 1.List接口常用方法1.代码2.结果 2.ArrayList底层机制1.结论2.ArrayList底层源码1.代码2.debug添加第一个元素1.进入2.elementData数组存储ArrayList的数据3.初始化为空数组4.首先确保使用size1来计算最小容量5.如果elementData为空&#xff0c;最小容量就是106.modCo…

Python爬虫(一)- Requests 安装与基本使用教程

文章目录 前言一、简介及安装1. 简介2. 安装 Requests2.1 安装2.2 检查安装是否成功 二、使用 Requests 发送 HTTP 请求1. 发送 GET 请求2. 发送 POST 请求3. 发送 PUT 请求4. 发送 DELETE 请求5. 发送 HEAD 请求6. 发送 OPTIONS 请求 三、传递参数1. GET 请求传递 URL 参数1.1…

风力涡轮机缺陷检测数据集,86.6%准确识别率,11921张图片,支持yolo,PASICAL VOC XML,COCO JSON格式的标注

风力涡轮机缺陷检测数据集&#xff0c;86.6&#xff05;准确识别率&#xff0c;11921张图片&#xff0c;支持yolo&#xff0c;PASICAL VOC XML&#xff0c;COCO JSON格式的标注 数据集下载 yolov11&#xff1a; https://download.csdn.net/download/pbymw8iwm/90206849 yolov…