分页
介绍
在其他框架中,分页可能非常痛苦。Laravel 让它变得轻而易举。Laravel 可以快速生成基于当前页面的智能“范围”链接,并且生成的 HTML 与 Bootstrap CSS 框架 兼容。
基本用法
分页查询构建器结果
有几种方法可以分页项目。最简单的方法是使用 查询构建器 或 Eloquent 查询 上的 paginate
方法。Laravel 提供的 paginate
方法会自动根据用户正在查看的当前页面设置适当的限制和偏移量。默认情况下,当前页面由 HTTP 请求上的 ?page
查询字符串参数的值检测。当然,这个值是由 Laravel 自动检测的,并且也会自动插入到分页器生成的链接中。
首先,让我们看看在查询上调用 paginate
方法。在此示例中,传递给 paginate
的唯一参数是您希望每页显示的项目数。在这种情况下,我们指定希望每页显示 15
个项目:
<?php
namespace App\Http\Controllers;
use DB;
use App\Http\Controllers\Controller;
class UserController extends Controller
{
/**
* 显示应用程序的所有用户。
*
* @return Response
*/
public function index()
{
$users = DB::table('users')->paginate(15);
return view('user.index', ['users' => $users]);
}
}
NOTE
目前,使用 groupBy
语句的分页操作无法由 Laravel 高效执行。如果您需要使用 groupBy
进行分页结果集,建议您查询数据库并手动创建分页器。
“简单分页”
如果您只需要在分页视图中显示简单的“下一页”和“上一页”链接,您可以选择使用 simplePaginate
方法来执行更高效的查询。如果您不需要在渲染视图时显示每个页码的链接,这对于大型数据集非常有用:
$users = DB::table('users')->simplePaginate(15);
分页 Eloquent 结果
您也可以分页 Eloquent 查询。在此示例中,我们将以每页 15
个项目的方式分页 User
模型。正如您所见,语法几乎与分页查询构建器结果相同:
$users = App\User::paginate(15);
当然,您可以在查询上设置其他约束后调用 paginate
,例如 where
子句:
$users = User::where('votes', '>', 100)->paginate(15);
您也可以在分页 Eloquent 模型时使用 simplePaginate
方法:
$users = User::where('votes', '>', 100)->simplePaginate(15);
手动创建分页器
有时您可能希望手动创建分页实例,并传递一个项目数组。您可以通过创建 Illuminate\Pagination\Paginator
或 Illuminate\Pagination\LengthAwarePaginator
实例来实现,具体取决于您的需求。
Paginator
类不需要知道结果集中的项目总数;然而,由于这个原因,该类没有用于检索最后一页索引的方法。LengthAwarePaginator
接受几乎与 Paginator
相同的参数;然而,它确实需要结果集中的项目总数。
换句话说,Paginator
对应于查询构建器和 Eloquent 上的 simplePaginate
方法,而 LengthAwarePaginator
对应于 paginate
方法。
在手动创建分页器实例时,您应该手动“切片”传递给分页器的结果数组。如果您不确定如何执行此操作,请查看 array_slice PHP 函数。
在视图中显示结果
当您在查询构建器或 Eloquent 查询上调用 paginate
或 simplePaginate
方法时,您将收到一个分页器实例。调用 paginate
方法时,您将收到 Illuminate\Pagination\LengthAwarePaginator
的实例。调用 simplePaginate
方法时,您将收到 Illuminate\Pagination\Paginator
的实例。这些对象提供了几个描述结果集的方法。除了这些辅助方法外,分页器实例是迭代器,可以像数组一样循环。
因此,一旦您检索到结果,您可以使用 Blade 显示结果并渲染页面链接:
<div class="container">
@foreach ($users as $user)
{{ $user->name }}
@endforeach
</div>
{{ $users->links() }}
links
方法将渲染结果集中其余页面的链接。每个链接都将包含正确的 ?page
查询字符串变量。请记住,links
方法生成的 HTML 与 Bootstrap CSS 框架 兼容。
自定义分页器 URI
setPath
方法允许您自定义分页器在生成链接时使用的 URI。例如,如果您希望分页器生成类似 http://example.com/custom/url?page=N
的链接,您应该将 custom/url
传递给 setPath
方法:
Route::get('users', function () {
$users = App\User::paginate(15);
$users->setPath('custom/url');
//
});
附加到分页链接
您可以使用 appends
方法将查询字符串添加到分页链接。例如,要将 &sort=votes
附加到每个分页链接,您应该对 appends
进行以下调用:
{{ $users->appends(['sort' => 'votes'])->links() }}
如果您希望将“哈希片段”附加到分页器的 URL,您可以使用 fragment
方法。例如,要将 #foo
附加到每个分页链接的末尾,请对 fragment
方法进行以下调用:
{{ $users->fragment('foo')->links() }}
其他辅助方法
您还可以通过分页器实例上的以下方法访问其他分页信息:
$results->count()
$results->currentPage()
$results->firstItem()
$results->hasMorePages()
$results->lastItem()
$results->lastPage() (使用 simplePaginate 时不可用)
$results->nextPageUrl()
$results->perPage()
$results->previousPageUrl()
$results->total() (使用 simplePaginate 时不可用)
$results->url($page)
将结果转换为 JSON
Laravel 分页器结果类实现了 Illuminate\Contracts\Support\JsonableInterface
合约,并公开了 toJson
方法,因此将分页结果转换为 JSON 非常简单。
您还可以通过简单地从路由或控制器操作返回分页器实例来将其转换为 JSON:
Route::get('users', function () {
return App\User::paginate();
});
分页器生成的 JSON 将包括诸如 total
、current_page
、last_page
等元信息。实际的结果对象将通过 JSON 数组中的 data
键提供。以下是从路由返回分页器实例创建的 JSON 示例:
分页器 JSON 示例
{
"total": 50,
"per_page": 15,
"current_page": 1,
"last_page": 4,
"next_page_url": "http://laravel.app?page=2",
"prev_page_url": null,
"from": 1,
"to": 15,
"data":[
{
// 结果对象
},
{
// 结果对象
}
]
}