Java项目开发中实现分页的三种方式一篇包会
前言
Java项目开发中经常要用到分页功能,现在普遍使用SpringBoot进行快速开发,而数据层主要整合SpringDataJPA和MyBatis两种框架,这两种框架都提供了相应的分页工具,使用方式也很简单,可本人在工作中除此以外还用到第三种更方便灵活的分页方式,在这里一同分享给大家。
使用
主要分为SpringDataJPA分页、MyBatis分页、Hutools工具类分页几个部分
1、SpringDataJPA分页
1)、引入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
2)、Service中编写分页服务
SpringDataJPA分页就是定义Pageable对象来处理分页,其中PageRequest来定义分页参数,Page对象来接手查询结果进行分页包装,包装后的结果pageResult可以得到总记录数、总页数、分页列表等数据结果。
/**
* 根据doctorId查询全部关注列表【分页】
*
* @param doctorId 医生id
* @return 结果集
*/
public Map<String, Object> findAllListByDoctorId(Long doctorId, Integer pageIndex, Integer pageSize) {
Pageable pageable = PageRequest.of(pageIndex - 1, pageSize); // 分页
Page<Follow> pageResult = followRepository.findByDoctorIdOrderByCreatedAtDesc(doctorId, pageable);
List<FollowDTO> dtoList = followMapper.toDto(pageResult.getContent());
if (!CollectionUtils.isEmpty(dtoList)) {
// 处理业务逻辑....
}
// 封装分页结果
Map<String, Object> map = new HashMap<>();
map.put("pageIndex", pageIndex.toString()); // 当前页
map.put("pageSize", pageSize.toString()); // 每页条数
map.put("total", Long.toString(pageResult.getTotalElements())); // 总记录数
map.put("pages", Integer.toString(pageResult.getTotalPages())); // 总页数
map.put("list", dtoList); // 数据列表
return map;
}
3)、Repository中处理分页
这里就是继承JpaRepository对象,然后传入service中定义好的pageable对象,并且返回Page包装的结果即可。
import com.patient.domain.Follow;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.jpa.repository.JpaRepository;
@Repository
public interface FollowRepository extends JpaRepository<Follow, Long> {
Page<Follow> findByDoctorIdOrderByCreatedAtDesc(Long doctorId, Pageable pageable);
}
2、MyBatis分页
1)、引入PageHelper依赖
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper-spring-boot-starter</artifactId>
</dependency>
2)、使用PageHelper实现分页
/**
* 查询推广人员列表,分页。
* @return 封装的分页结果对象
*/
public PageResult findPromotePersonList(String hospitalCode,PromotePersonReq promotePersonReq) {
Integer pageIndex = promotePersonReq.getPageIndex();
Integer pageSize = promotePersonReq.getPageSize();
PageHelper.startPage(pageIndex, pageSize); // 每页的大小为pageSize,查询第page页的结果
List<PromotePerson> list = promotePersonMapper.selectAll();
PageInfo<PromotePerson> pageInfo = new PageInfo<>(list);
PageHelper.clearPage();
// 返回分页结果
PageResult pageResult = new PageResult();
pageResult.setPageIndex(pageIndex);
pageResult.setPageSize(pageSize);
pageResult.setPages(pageInfo.getPages());
pageResult.setTotal((int) pageInfo.getTotal());
pageResult.setList(list);
return pageResult;
}
3、Hutools工具类分页
1)、引入依赖
这里是可以单独引入hutools部分工具类的,具体参考官网文档,我平时写项目一定会使用这个工具,所以直接引入了所有。
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.1.2</version>
</dependency>
2)、分页实现
一般就用到两个工具类,一是PageUtil.totalPage(总记录数, 每页记录数)来计算总页数,二是CollUtil.page(索引, 每页记录数, 数据列表)来返回指定分页结果,注意这里的索引是从1开始的,和SpringDataJPA分页索引从0开始有区别。
/**
* 我的订单-待支付[分页]
*
* @param openid 用户唯一标识
* @return 结果集
*/
public Map<String, Object> findMyOrderNotPay(String openid, Integer pageIndex, Integer pageSize) {
Map<String, Object> map = new HashMap<>();
// 查询
List<ConsultOrder> orderList = consultOrderRepository.findMyOrderNotPay(openid);
if (CollectionUtils.isEmpty(orderList)) {
map.put("pageIndex", pageIndex.toString()); // 当前页
map.put("pageSize", pageSize.toString()); // 每页条数
map.put("total", "0"); // 总记录数
map.put("pages", "0"); // 总页数
map.put("list", new ArrayList<>()); // 数据列表
return map;
}
List<OrderVO> pageList = new ArrayList<>();
int totalSize = 0;
int totalPage = 0;
// 计算总页数
totalSize = orderList.size();
totalPage = PageUtil.totalPage(totalSize, pageSize);
// 分页,索引小于等于总页数,才返回列表.
if (pageIndex <= totalPage) {
// 分页
pageList = CollUtil.page(pageIndex, pageSize, orderVOList);
}
// 返回结果
map.put("pageIndex", Integer.toString(pageIndex)); // 当前页
map.put("pageSize", Integer.toString(pageSize)); // 每页条数
map.put("total", Integer.toString(totalSize)); // 总记录数
map.put("pages", Integer.toString(totalPage)); // 总页数
map.put("list", pageList); // 数据列表
return map;
}
总结
1)、注意 :为了方便演示代码中直接用了Map对象来包装返回分页结果,在实际项目中,切记一定要自己定义实体对象作为返回结果,因为Map对象返回的结果如果是动态且数据量较大的列表,是存在造成内存泄露风险的,举个例子,比如返回10条问诊的分页记录时,其中聊天内容这个属性包含大量聊天数据,因为你无法确定对话的两个人到底聊了多少,可它确实作为一个属性包含在分页记录中;
2)、SpringDataJPA分页,就是使用自带的Pageable对象来处理,需要注意的是分页索引从0开始,传错了会造成分页结果错乱或重复的现象;
3)、Mybatis分页,就是借助PageHelper工具来实现,PageHelper.startPage和PageHelper.clearPage中间是需要分页的业务查询代码,可以通过PageInfo对象包装,获取其中需要的分页参数返回给前端展示;
4)、Hutools分页,就是引入hutools工具类,使用其中的PageUtil和CollUtil工具类来实现,这种方式我个人比较喜欢,因为在较复杂的查询业务中,前两种实现起来很费劲还容易写错,不仅可能牵扯到多个类及方法,写完后过段时间也不容易阅读。而Hutools分页就类似于很早以前的分页方式,我把它理解为绿色简易版JSP分页,只需在服务层使用一个工具类分页即可,既灵活又便于阅读,简直是分页的神器。
本人文章从来都是纯手打,且都来自实际工作中的经验及分享,如果觉得有一滴滴帮助,就点个推荐吧!(o..o)~
Java项目开发中实现分页的三种方式一篇包会的更多相关文章
- 《Maven在Java项目开发中的应用》论文笔记(十七)
标题:Maven在Java项目开发中的应用 一.基本信息 时间:2019 来源:山西农业大学 关键词:Maven:Java Web:仓库:开发人员:极限编程; 二.研究内容 1.Maven 基本原理概 ...
- Java反射机制(创建Class对象的三种方式)
1:SUN提供的反射机制的类: java.lang.Class<T> java.lang.reflect.Constructor<T> java.lang.reflect.Fi ...
- android中解析文件的三种方式
android中解析文件的三种方式 好久没有动手写点东西了,最近在研究android的相关技术,现在就android中解析文件的三种方式做以下总结.其主要有:SAX(Simple API fo ...
- Struts中的数据处理的三种方式
Struts中的数据处理的三种方式: public class DataAction extends ActionSupport{ @Override public String execute() ...
- iOS开发 跳转场景的三种方式
iOS开发 跳转场景的三种方式 2012年10月17日, 15:32 假设A跳转到B,三种方法:1.按住ctrl键,拖动A上的控件(比如说UIButton)到B上,弹出菜单,选择Modal.不需要写任 ...
- Java 数组元素逆序Reverse的三种方式
Java 数组元素逆序Reverse的三种方式 本文链接:https://blog.csdn.net/xHibiki/article/details/82930521 题目 代码实现 说明 int ...
- HTML中使用js的三种方式及优缺点介绍
1.内部js: 在直接在页面的<script></script>标签内写js代码 优点:相对于使用行内js,内部js代码较为集中,与页面结构的实现代码耦合度较低,比较便于维护 ...
- JS中事件绑定的三种方式
以下是搜集的在JS中事件绑定的三种方式. 1. HTML onclick attribute <button type="button" id="upl ...
- JavaScript 中事件绑定的三种方式
以下是在 JS 中事件绑定的三种方式. 1. HTML onclick attribute <button type="button" id="uplo ...
随机推荐
- JS常用的获取值和设值的方法
1. input 标签<input type="text" name="username" id="name"/> 1) 获取i ...
- 基于 Electron 的 Rubick 2.4k star 啦,同步更新新功能!
为什么要做 Rubick 其实做 Rubick 1.x 的初衷就是解决自己的问题的:特别需要一款支持自定义插件的桌面端应用来简化使用者安装庞大桌面端应用的臃肿.而且涉及到数据安全的问题,插件只能在公司 ...
- CS5218DP转HDMI转接方案|CS5218说明|CS5218
Capstone CS5218是一款单端口HDMI/DVI电平移位器/中继器,具有重新定时功能.它支持交流和直流耦合信号高达3.0-Gbps的操作与可编程均衡和抖动清洗.它包括2路双模DP电缆适配器寄 ...
- 云南农业职业技术学院 - 互联网技术学院 - 美和易思《MYSQL 高级查询与编程》 综合机试试卷
数据库及试题文档下载:https://download.csdn.net/download/weixin_44893902/14503097 目录 题目:电商平台 mysql 数据库系统管理 一. 语 ...
- Java初学者作业——编写JAVA程序,根据用户输入课程名称,输出对应课程的简介,各门课程的简介见表
返回本章节 返回作业目录 需求说明: 编写JAVA程序,根据用户输入课程名称,输出对应课程的简介,各门课程的简介见表 课程名称 课程简介 JAVA课程 JAVA语言是目前最流行的编写语言,在本课程中将 ...
- Ubuntu18.04安装/卸载NVIDIA显卡驱动
1 显卡驱动下载 官网:NVIDIA 搜索适合本机的驱动 获取最新版本驱动 立即下载 文件 以上,显卡驱动下载完成. 2 显卡驱动安装 2.1 添加可执行权限 进入驱动文件目录sudo chmod a ...
- CSS基础6之盒子模型1
盒子概述 以下是盒子模型的一个图形解释 一.内边距(填充) 属性有: (1) padding 设置所有内边距 (2) padding-top 设置上边距 (3) padding-left 设置左边 ...
- 不同目录存在相同名称的py文件,执行时,报错的解决方法
1.问题现象如下,执行时报错 imported module 'test_case_execution' has this __file__ attribute platform win32 -- ...
- 深入研究 synchronized 同步锁 作用于 静态方法 和 非静态方法 的 区别
1.前言 众所周知, synchronized 是同步锁 ,虽然在底层又细分了无锁.偏向锁.轻量级锁.自旋锁 以及重量级锁 机制, 这些底层锁知道一下原理即可 ,[想要 了解 这篇 博文 有 解释 : ...
- HDU-1004(C语言描述)
Let the Balloon Rise 输入 输入包含多个测试用例.每个测试用例都以数字 N (0 < N < = 1000) 为起点, 分布的气球总数.下 N 行包含一个颜色.气球的颜 ...