0

0

如何在 Laravel 中基于查询参数正确实现每页 20 条的数据分页

聖光之護

聖光之護

发布时间:2026-01-13 11:12:10

|

485人浏览过

|

来源于php中文网

原创

如何在 Laravel 中基于查询参数正确实现每页 20 条的数据分页

本文详解如何修复基于 `offset` 查询参数的手动分页逻辑错误,指出原始代码中页码与偏移量混淆的问题,并提供使用 laravel 集合 `chunk()` 方法及更健壮的 `forpage()` 方案。

你当前的分页逻辑存在一个关键概念性错误:将查询参数 offSet(应为 偏移量,即跳过的记录数)误当作 页码(page number)来使用。

在你的代码中:

$page = $request->get('offSet'); // ❌ 错误:offSet 是偏移量,不是页码
$take = 20;
$skip = ($page - 1) * $take; // 若 offSet=2,则 skip = 20 → 跳过前20条,取第21–40条 → 实际是第2页,但语义混乱!

这导致行为不可预测:当用户传 offSet=1 时,skip = 0,显示第 1–20 条(看似“第1页”);但传 offSet=2 时,skip = 20,显示第 21–40 条——这其实是按偏移量直接切片,而非标准分页。而 offSet=4 会跳过 60 条,若总数据不足 80 条,结果为空,因此“不工作”。

✅ 正确做法有两种推荐方案:

方案一:使用 forPage($page, $perPage)(推荐 ✅)

这是 Laravel 集合原生支持的标准分页方法,语义清晰、零越界风险:

Viggle AI
Viggle AI

Viggle AI是一个AI驱动的3D动画生成平台,可以帮助用户创建可控角色的3D动画视频。

下载
$results = collect($response->json()['results']);
$page = (int) $request->input('page', 1); // 默认第1页
$perPage = 20;

$paginated = $results->forPage($page, $perPage)->values()->all();
// ->values() 重置键名,确保返回连续数字索引数组
✅ 优势:自动处理页码越界(如请求第100页但只有50条数据,返回空集合)、无需手动计算 skip,符合 REST 分页惯例(?page=1, ?page=2)。

方案二:若坚持用 offset 参数(如兼容旧 API),请直译为偏移量

$offset = max(0, (int) $request->input('offSet', 0));
$limit = 20;
$paginated = $results->slice($offset, $limit)->values()->all();

⚠️ 注意:此时 offSet=0 → 第1页,offSet=20 → 第2页……需前端严格按 offset = (page-1)*20 构造 URL,否则语义断裂。

❌ 不推荐:chunk() + 数组访问

答案中提到的 $results->chunk(20)[$request->input('offset')] 存在严重隐患:

// 示例:$results 有 45 条 → chunk(20) 得到集合 [ [0..19], [20..39], [40..44] ]
// 若 $request->input('offset') == 3 → 尝试访问索引 3 → PHP Notice: Undefined offset

chunk() 返回的是「按块分组的集合」,其键是块序号(0-indexed),但 offset 参数通常代表偏移量(非块号),二者单位不同,强行映射极易越界崩溃。

最佳实践总结

  • ✅ 前端传参统一用 ?page=1&per_page=20,后端用 forPage();
  • ✅ 必须校验输入:(int) $request->input('page', 1) 防止非数字注入;
  • ✅ 总数未知时,可用 count() + forPage() 安全分页;若需分页元信息(总页数/总数),建议改用数据库分页(paginate());
  • ❌ 避免手动 skip/take 计算,除非你完全掌控数据规模且需极致性能。

修正后的完整示例:

$results = collect($response->json()['results']);
$page = max(1, (int) $request->input('page', 1));
$perPage = 20;

$items = $results->forPage($page, $perPage)->values()->all();
return response()->json([
    'data' => $items,
    'meta' => [
        'current_page' => $page,
        'per_page' => $perPage,
        'total' => $results->count(),
        'last_page' => (int) ceil($results->count() / $perPage),
    ]
]);

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

316

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

271

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

368

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

368

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

81

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

64

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.08.05

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.11.20

PHP 表单处理与文件上传安全实战
PHP 表单处理与文件上传安全实战

本专题聚焦 PHP 在表单处理与文件上传场景中的实战与安全问题,系统讲解表单数据获取与校验、XSS 与 CSRF 防护、文件类型与大小限制、上传目录安全配置、恶意文件识别以及常见安全漏洞的防范策略。通过贴近真实业务的案例,帮助学习者掌握 安全、规范地处理用户输入与文件上传的完整开发流程。

5

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 8.5万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 6.9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号