Laravel --實戰篇 (自定義分頁類)

前言 : Laravel框架中有封裝好的分頁類paginate方法,可是這個方法若是是在數據量比較大的項目中,無疑是在無形中給項目挖了一個隱藏的坑,由於若是是看DB類源碼paginate的分頁基礎是把全部的數據都拿了,只不過根據nowPage給你分頁的數據罷了,並且該分頁類,你作不了緩存處理。php

下面給你們分享一個乾貨,經過修改源碼封裝出來的一個自定義分頁類,而且該自定義分頁類,支持Redis緩存覆蓋。(本篇文章中暫不介紹Redis方面)前端

1.新的獲取數據的方法

獲取數據的方法有兩種(其中一種是手冊裏面沒有介紹,經過讀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();

這種方法比較便捷直接傳一個當前頁和具體每頁多少條數據就好了,以上兩種方法取的數據是如出一轍的。函數

2.計算總數據條數

$count = DB::table('user_master')->count();

3.計算總頁數

$countPage = ceil($count / $pageNum);

用進一取整方法獲取總頁數。優化

4.調用自定義分頁類(正在的乾貨立刻要出現了)

$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數據的分頁類,請繼續關注,後面將會講到部分項目優化

相關文章
相關標籤/搜索