Spring Data JPA 中的分页实现:从 BasePage 到 Pageable

文章目录

  • Spring Data JPA 中的分页实现:从 BasePage 到 Pageable
    • 背景:为什么需要分页?
    • 认识 BasePage 类
    • 深入 toPageable() 方法
      • 1. 处理页码和页面大小
      • 2. 处理排序方向
      • 3. 处理排序字段
      • 4. 生成 Pageable 对象
    • 实战:如何使用 BasePage
      • 1. 前端请求
      • 2. 后端控制器
      • 3. 查询数据库
      • 4. 返回结果
    • 小结


Spring Data JPA 中的分页实现:从 BasePage 到 Pageable

在现代 Web 开发中,分页是一个常见的需求。无论是展示商品列表、文章列表还是用户数据,当数据量较大时,分页不仅能提升用户体验,还能减轻服务器的压力。如果你使用的是 Spring Boot 和 Spring Data JPA,那么分页功能已经内置其中,开发者只需简单配置即可实现。今天,我们通过一个简单的 BasePage 类,来看看如何优雅地实现分页和排序功能。

背景:为什么需要分页?

假设你正在开发一个电商平台,数据库中有成千上万的商品记录。如果一次性将所有数据返回给前端,不仅加载速度会变慢,用户也很难从中找到所需信息。这时,分页就派上用场了:每页显示固定数量的记录(比如 10 条),用户可以通过翻页操作逐步浏览。

Spring Data JPA 提供了一个强大的接口 Pageable,它封装了分页和排序的逻辑。通过 Pageable,我们可以轻松实现“第几页”、“每页几条”以及“按什么字段排序”的功能。而今天的主角——BasePage 类,就是一个自定义的分页模型,通过它我们可以将前端传入的分页参数转换为 Pageable 对象。

认识 BasePage 类

BasePage 是一个简单的分页模型类,包含了分页和排序的核心属性。以下是它的代码结构:

public class BasePage {@ApiModelProperty("页码")Integer page;@ApiModelProperty("页大小")Integer size;@ApiModelProperty(value = "排序规则(ASC/DESC)", allowableValues = "ASC,DESC")String direction;@ApiModelProperty("排序字段")String[] properties;// 构造函数、getter 和 setter 略public Pageable toPageable() {page = page != null ? page : 0;size = size != null ? size : 9999;Sort.Direction dir = Sort.Direction.fromOptionalString(this.direction).orElse(Sort.Direction.DESC);Sort sort = (properties != null && properties.length > 0) ? Sort.by(dir, properties) : Sort.by(dir, "createdDate");return PageRequest.of(page, size, sort);}
}

这个类包含了四个关键属性:

  • page:页码,表示当前是第几页。
  • size:每页的大小,表示每页显示多少条记录。
  • direction:排序方向,可以是 "ASC"(升序)或 "DESC"(降序)。
  • properties:排序字段,表示按哪些字段排序。

toPageable() 方法则是这个类的核心,它将这些属性转化为 Spring Data JPA 所需的 Pageable 对象。

深入 toPageable() 方法

让我们一步步拆解 toPageable() 方法,看看它是如何工作的:

1. 处理页码和页面大小

page = page != null ? page : 0;
size = size != null ? size : 9999;
  • 如果 page 未设置(null),默认值为 0,表示第一页(Spring Data 的页码从 0 开始)。
  • 如果 size 未设置,默认值为 9999。这是一个很大的数字,通常是为了在未指定页面大小时尽量返回所有数据。不过在实际应用中,建议根据业务需求设置一个合理的默认值,比如 10 或 20。

2. 处理排序方向

Sort.Direction dir = Sort.Direction.fromOptionalString(this.direction).orElse(Sort.Direction.DESC);
  • Sort.Direction 是 Spring Data 的枚举类,表示排序方向。
  • fromOptionalString(this.direction) 尝试将 direction(如 “ASC” 或 “DESC”)解析为 Sort.Direction 类型。
  • 如果解析失败或 directionnull,则使用默认值 Sort.Direction.DESC(降序)。

3. 处理排序字段

Sort sort = (properties != null && properties.length > 0) ? Sort.by(dir, properties) : Sort.by(dir, "createdDate");
  • 如果 properties 不为空,则按指定的字段排序。
  • 如果 properties 为空或未设置,则默认按 "createdDate"(创建时间)字段排序。
  • Sort.by(dir, properties) 创建了一个 Sort 对象,封装了排序方向和字段。

4. 生成 Pageable 对象

return PageRequest.of(page, size, sort);
  • 使用 PageRequest.of() 方法,将页码、页面大小和排序规则组合成一个 Pageable 对象。
  • 这个对象可以直接传递给 Spring Data 的查询方法。

实战:如何使用 BasePage

假设我们有一个商品管理的接口,需要支持分页和排序。以下是实现步骤:

1. 前端请求

前端发送一个 HTTP 请求,例如:

GET /products?page=1&size=10&direction=ASC&properties=price
  • page=1:请求第 2 页(从 0 开始计数)。
  • size=10:每页 10 条记录。
  • direction=ASC:按升序排序。
  • properties=price:按价格排序。

2. 后端控制器

在 Spring Boot 的控制器中接收参数并构造 BasePage

@RestController
@RequestMapping("/products")
public class ProductController {@Autowiredprivate ProductRepository productRepository;@GetMappingpublic Page<Product> getProducts(@RequestParam(required = false) Integer page,@RequestParam(required = false) Integer size,@RequestParam(required = false) String direction,@RequestParam(required = false) String[] properties) {BasePage basePage = new BasePage();basePage.setPage(page);basePage.setSize(size);basePage.setDirection(direction);basePage.setProperties(properties);Pageable pageable = basePage.toPageable();return productRepository.findAll(pageable);}
}

3. 查询数据库

ProductRepository 是一个 Spring Data JPA 提供的接口:

public interface ProductRepository extends JpaRepository<Product, Long> {
}

调用 findAll(Pageable pageable) 方法后,Spring Data 会自动生成 SQL 查询,返回分页后的结果。

4. 返回结果

最终返回的 JSON 可能长这样:

{"content": [{"id": 1, "name": "商品A", "price": 10.0},{"id": 2, "name": "商品B", "price": 15.0},...],"totalElements": 50,"totalPages": 5,"number": 1,"size": 10
}

小结

通过 BasePagetoPageable() 方法,我们实现了一个简单而灵活的分页模型。它的优点在于:

  • 简单易用:只需设置几个属性,就能生成 Pageable 对象。
  • 默认值友好:未设置的参数有合理的默认值,避免空指针问题。
  • 支持排序:灵活指定排序字段和方向,满足复杂需求。

当然,实际开发中还可以根据需求扩展 BasePage,比如添加参数验证、支持多字段排序等。Spring Data JPA 的分页功能非常强大,配合自定义模型类,可以让代码更优雅、更易维护。

如果你也正在为分页功能苦恼,不妨试试这种方式吧!有什么问题或想法,欢迎在评论区交流。


在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android SystemUI开发(一)

frameworks/base/packages/SystemUI/src/com/android/systemui/SystemUI.java frameworks/base/packages/SystemUI/src/com/android/systemui/SystemUIService.java 关键文件 SystemUI 关键服务 简介 Dependency.class&#xff1a;处理系统依赖关系&#xff0c;提供资源或服…

Python----Python爬虫(多线程,多进程,协程爬虫)

注意&#xff1a; 该代码爬取小说不久或许会失效&#xff0c;有时候该网站会被封禁&#xff0c;代码只供参考&#xff0c;不同小说不同网址会有差异 神印王座II皓月当空最新章节_神印王座II皓月当空全文免费阅读-笔趣阁 一、多线程爬虫 1.1、单线程爬虫的问题 爬虫通常被认为…

Linux(ftrace)__mcount的实现原理

Linux 内核调试工具ftrace 之&#xff08;_mcount的实现原理&#xff09; ftrace 是 Linux 内核中的一种跟踪工具&#xff0c;主要用于性能分析、调试和内核代码的执行跟踪。它通过在内核代码的关键点插入探针&#xff08;probe&#xff09;来记录函数调用和执行信息。这对于开…

网络层IP协议

基本概念 主机&#xff1a;有IP地址&#xff0c;但是不进行路由控制的设备。 路由器&#xff1a;有IP地址&#xff0c;又能进行路由控制。 节点&#xff1a;主机和路由器的统称。 协议头格式 4位版本号&#xff1a;指定IP协议的版本。对于IPv4来说&#xff0c;就是4 4位首…

计算机毕业设计SpringBoot+Vue.js在线课程管理系统(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

【腾讯云】AI驱动TDSQL-C Serveress 数据库技术实战营-如何是从0到1体验电商可视化分析小助手得统计功能,一句话就能输出目标统计图

欢迎来到《小5讲堂》 这是《腾讯云》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 背景效果图流程图创建数据库 基本信息数据库配置设置密码控制台开…

Java里的ArrayList和LinkedList有什么区别?

大家好&#xff0c;我是锋哥。今天分享关于【Java里的ArrayList和LinkedList有什么区别&#xff1f;】面试题。希望对大家有帮助&#xff1b; Java里的ArrayList和LinkedList有什么区别&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 ArrayList 和 Lin…

盛京开源社区加入 GitCode,书写东北开源生态新篇章

在数字化转型与开源技术蓬勃发展的浪潮下&#xff0c;开源社区已成为推动技术创新的核心力量。盛京开源社区&#xff08;SJOSC&#xff09;作为沈阳地区的开源交流平台&#xff0c;始终致力于连接开发者、企业及高校&#xff0c;构建区域技术生态圈。 现在&#xff0c;盛京开源…

SQL Server 创建用户并授权

创建用户前需要有一个数据库&#xff0c;创建数据库命令如下&#xff1a; CREATE DATABASE [数据库名称]; CREATE DATABASE database1;一、创建登录用户 方式1&#xff1a;SQL命令 命令格式&#xff1a;CREATE LOGIN [用户名] WITH PASSWORD ‘密码’; 例如&#xff0c;创…

vue3:三项目增加404页面

一、路由添加 1、官网地址 带参数的动态路由匹配 | Vue Routerhttps://router.vuejs.org/zh/guide/essentials/dynamic-matching.html 2、复制核心语句 { path: /:pathMatch(.*)*, name: NotFound, component: NotFound } 3、粘贴到路由index.js中 4、建立页面 在view文件夹…

GitCode 助力 JeeSite:开启企业级快速开发新篇章

项目仓库&#xff08;点击阅读原文链接可直达前端仓库&#xff09; https://gitcode.com/thinkgem/jeesite 企业级快速开发的得力助手&#xff1a;JeeSite 快速开发平台 JeeSite 不仅仅是一个普通的后台开发框架&#xff0c;而是一套全面的企业级快速开发解决方案。后端基于 …

矽电股份业绩下滑:毛利率也欠佳,应收账款攀升回款比率放缓

《港湾商业观察》施子夫 近期&#xff0c;矽电半导体设备&#xff08;深圳&#xff09;股份有限公司&#xff08;以下简称&#xff0c;矽电股份&#xff09;的深交所IPO注册获得生效。 公开信息显示&#xff0c;2022年6月&#xff0c;矽电股份的创业板IPO获受理&#xff0c;保…

MySQL实现文档全文搜索,分词匹配多段落重排展示,知识库搜索原理分享

一、背景 在文档搜索场景中&#xff0c;高效精准的搜索功能至关重要&#xff0c;能提升检索效率&#xff0c;为用户提供精准、快速的信息获取体验&#xff0c;提高工作效率。在文档管理系统里&#xff0c;全文搜索是非常重要的功能之一。随着文档数量增长&#xff0c;如何快速…

十、大数据资源平台功能架构

一、大数据资源平台的功能架构图总体结构 大数据资源平台功能架构图 关键组件&#xff1a; 1.用户&#xff08;顶行&#xff09; 此部分标识与平台交互的各种利益相关者。 其中包括&#xff1a; 市领导 各部门分析师 区政府 外部组织 公民 开发人员 运营经理 2.功能模…

现代前端框架渲染机制深度解析:虚拟DOM到编译时优化

引言&#xff1a;前端框架的性能进化论 TikTok Web将React 18迁移至Vue 3后&#xff0c;点击响应延迟降低42%&#xff0c;内存占用减少35%。Shopify采用Svelte重构核心交互模块&#xff0c;首帧渲染速度提升580%。Discord在Next.js 14中启用React Server Components后&#xf…

【子网掩码计算器:Python + Tkinter 实现】

子网掩码计算器&#xff1a;Python Tkinter 实现 引言代码功能概述代码实现思路1. 界面设计2. 功能实现3. 事件处理 子网掩码计算器实现步骤1. 导入必要的库2. 定义主窗口类 SubnetCalculatorApp3. 创建菜单栏4. 创建界面组件5. 判断 IP 地址类别6. 计算子网信息7. 其他功能函…

视频推拉流EasyDSS点播平台云端录像播放异常问题的排查与解决

EasyDSS视频直播点播平台是一个功能全面的系统&#xff0c;提供视频转码、点播、直播、视频推拉流以及H.265视频播放等一站式服务。该平台与RTMP高清摄像头配合使用&#xff0c;能够接收无人机设备的实时视频流&#xff0c;实现无人机视频推流直播和巡检等多种应用。 最近&…

android 文本控件显示滑动条并自动滑动到最底部

文本框滑动需要增加控件 设置属性显示滑动条垂直滑动 显示滑动条 自动滚动到最后一行&#xff1a; private ScrollView mscrollView; mOutput.setMovementMethod(ScrollingMovementMethod.getInstance()); mscrollView (ScrollView) findViewById(R.id.scrollview); mscrol…

#渗透测试#批量漏洞挖掘#某图创图书馆集群管理系统updOpuserPw SQL注入(CVE-2021-44321)

免责声明 本教程仅为合法的教学目的而准备&#xff0c;严禁用于任何形式的违法犯罪活动及其他商业行为&#xff0c;在使用本教程前&#xff0c;您应确保该行为符合当地的法律法规&#xff0c;继续阅读即表示您需自行承担所有操作的后果&#xff0c;如有异议&#xff0c;请立即停…

1.C语言初识

C语言初识 C语言初识基础知识hello world数据类型变量、常量变量命名变量分类变量的使用变量的作用域 常量字符字符串转义字符 选择语句循环语句 函数&#xff1b;数组函数数组数组下标 操作符操作符算术操作符移位操作符、位操作符赋值操作符单目操作符关系操作符逻辑操作符条…