前言 :
Laravel
框架中有封裝好的分頁類paginate
方法,可是這個方法若是是在數據量比較大的項目中,無疑是在無形中給項目挖了一個隱藏的坑,由於若是是看DB類源碼paginate
的分頁基礎是把全部的數據都拿了,只不過根據nowPage
給你分頁的數據罷了,並且該分頁類,你作不了緩存處理。php
下面給你們分享一個乾貨,經過修改源碼封裝出來的一個自定義分頁類,而且該自定義分頁類,支持Redis
緩存覆蓋。(本篇文章中暫不介紹Redis方面)前端
獲取數據的方法有兩種(其中一種是手冊裏面沒有介紹,經過讀Laravel能夠了解到)數組
(1)經過偏移量獲取緩存
$data = DB::table('user_master')->skip($start)->take($pagenum)->orderBy('time', 'desc')->get();
這種取發比較麻煩 須要經過每次傳過來的page
來計算偏移量。app
(2)經過DB類的forPage
方法獲取數據框架
$data = DB::table('user_master')->forPage($nowPage, $pageNum)->orderBy('time', 'desc')->get();
這種方法比較便捷直接傳一個當前頁和具體每頁多少條數據就好了,以上兩種方法取的數據是如出一轍的。函數
$count = DB::table('user_master')->count();
$countPage = ceil($count / $pageNum);
用進一取整方法獲取總頁數。優化
$pages = CustomPage::getSelfPageView($nowPage, $countPage, '/user', '');
介紹一下每一個參數 第1個參數:表示當前哪一頁,第二參數:總頁數,第三參數:頁面路由,第四參數:須要維持的參數(好比status=1);ui
下面就是乾貨時間了 那麼這個分頁類究竟是什麼呢。url
<?php namespace App\Tools; /** * 自定義分頁類,主要用於產生分頁試圖 * Class Common * @package App\Library */ class CustomPage { /** * @param $text * @return string */ public static function getActivePageWrapper($text) { return '<li><span>' . $text . '</span></li>'; } /** * 獲取當前頁按鈕的頁面樣式 * @param $url * @param $page * @return string */ public static function getActivePageLinkWrapper($url, $page) { return '<li class="active"><a href="' . $url . '">' . $page . '</a></li>'; } /** * 獲取非當前頁按鈕的頁面樣式 * @param $url * @param $page * @return string */ public static function getPageLinkWrapper($url, $page) { return '<li><a href="' . $url . '">' . $page . '</a></li>'; } /** * 獲取整個的分頁樣式 * @param $nowPage 當前頁 * @param $totalPage 共多少頁面 * @param $baseUrl 當前url * @param $search 搜索 * @return string */ public static function getSelfPageView($nowPage, $totalPage, $baseUrl, $search) { $pagePre = '<ul class="pagination">'; $pageEnd = '</ul>'; $pageLastStr = ''; $pageNextStr = ''; if ($nowPage <= 1) { $nowPage = 1; $pageLastStr = '<li class="disabled"><span>«</span></li>'; } if ($nowPage >= $totalPage) { $nowPage = $totalPage; $pageNextStr = '<li class="disabled"><span>»</span></li>'; } $search['totalPage'] = $totalPage; if (empty($pageLastStr)) { $lastPage = $nowPage - 1; $search['nowPage'] = $lastPage; $lastSearchStr = self::arrayToSearchStr($search); $url = $baseUrl . '?' . $lastSearchStr; $pageLastStr = self::getPageLinkWrapper($url, '«'); } if (empty($pageNextStr)) { $pageNext = $nowPage + 1; $search['nowPage'] = $pageNext; $lastSearchStr = self::arrayToSearchStr($search); $url = $baseUrl . '?' . $lastSearchStr; $pageNextStr = self::getPageLinkWrapper($url, '»'); } $pageTemp = ''; $pageRange = self::getPageRange($nowPage, $totalPage); $pageTemp .= $pageLastStr; foreach ($pageRange as $page) { $search['nowPage'] = $page; $searchStr = self::arrayToSearchStr($search); $url = $baseUrl . '?' . $searchStr; if ($page == $nowPage) { $pageTemp .= self::getActivePageLinkWrapper($url, $page); } else { $pageTemp .= self::getPageLinkWrapper($url, $page); } } $pageTemp .= $pageNextStr; $pageView = $pagePre . $pageTemp . $pageEnd; return $pageView; } /** * 獲取實際顯示頁面範圍的範圍 * @param $nowPage * @param $totalPage * @return array */ public static function getPageRange($nowPage, $totalPage) { $returnArray = []; if ($totalPage <= 5) { for ($i = 1; $i <= $totalPage; $i++) { $returnArray[] = $i; } } else { $lengthLeft = $nowPage - 1; $lengthRight = $totalPage - $nowPage; if (($lengthLeft < 2) && ($lengthRight < 2)) { $returnArray = []; } elseif (($lengthLeft < 2) && ($lengthRight > 2)) { for ($i = 1; $i <= 5; $i++) { $returnArray[] = $i; } } elseif (($lengthLeft > 2) && ($lengthRight < 2)) { $start = $totalPage - 4; for ($i = $start; $i <= $totalPage; $i++) { $returnArray[] = $i; } } else { for ($i = $nowPage - 2; $i <= $nowPage + 2; $i++) { $returnArray[] = $i; } } } return $returnArray; } /** * 將搜索的數組拼接成爲url * 注意:PHP的內置函數http_build_query,會自動將沒有值的參數清除,致使blade模板報錯 * @param $array * @return string */ public static function arrayToSearchStr($array) { $fields_string = ''; reset($array); end($array); $lastKey = key($array); reset($array); foreach ($array as $key => $value) { if ($key != $lastKey) { $fields_string .= $key . '=' . $value . '&'; } else { $fields_string .= $key . '=' . $value; } } rtrim($fields_string, '&'); return $fields_string; } }
記得使用分頁類時引入命名空間,最後在視圖中使用分頁類的方法與Laravel自帶的分頁類的方法是同樣的。
{!! $pages !!}
如何用Redis
覆蓋整個項目的思想(包括列表,後臺頁面,前端頁面)將在博主後面的博客中一一展示,其中Redis三部曲中就有一部分講到如何Redis代替MySQL數據的分頁類,請繼續關注,後面將會講到部分項目優化