java简单分页实现


BaseQuery.java
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonInclude;

import java.io.Serializable;

/**
 * @author yangqiaoxin
 * @date 2021/01/02
 */
@JsonIgnoreProperties(value = {"handler","hibernateLazyInitializer","fieldHandler"})
@JsonInclude(JsonInclude.Include.NON_NULL)
public abstract class BaseQuery implements Serializable {
}


PageInfo.java
/**
 * 分页信息
 * @author yangqiaoxin
 * @date 2021/01/01
 */
public class PageInfo<T> extends BaseQuery {

    /**
     * 当前页
     */
    private long pageNum;

    /**
     * 每页的数量
     */
    private long pageSize;

    /**
     * 当前页的数量
     */
    private long size;

    /**
     * 总记录数
     */
    private long total;

    /**
     * 总页数
     */
    private long pages;

    /**
     * 结果集
     */
    private List<T> list;

    /**
     * 第一页的页码
     */
    private long firstPage;

    /**
     * 最后一页的页码
     */
    private long lastPage;

    /**
     * 上一页的页码
     */
    private long prevPage;

    /**
     * 下一页的页码
     */
    private long nextPage;

    /**
     * 是否为第一页
     */
    private boolean hasFirstPage = false;

    /**
     * 是否为最后一页
     */
    private boolean hasLastPage = false;

    /**
     * 是否有上一页
     */
    private boolean hasPreviousPage = false;

    /**
     * 是否有下一页
     */
    private boolean hasNextPage = false;

    /**
     * 导航页码数
     */
    private long navigatePages;

    /**
     * 所有导航页号
     */
    private long[] navigatePageNums;

    /**
     * 导航页号第一页的页码
     */
    private long navigateFirstPage;

    /**
     * 导航页号最后一页的页码
     */
    private long navigateLastPage;

    // 此处省略构造函数、Getter、Setter、toString等方法

}


PageQuery.java
/**
 * 分页查询对象
 * @author yangqiaoxin
 * @date 2021/01/01
 */
public class PageQuery extends BaseQuery {

    /**
     * 偏移量
     */
    private long offset;

    /**
     * 查询的记录数
     */
    private long size;

    // 此处省略构造函数、Getter、Setter、toString等方法

}


PageUtil.java
/**
 * 简单的分页工具类
 * @author yangqiaoxin
 * @date 2021/01/01
 */
public class PageUtil {

    /**
     * 默认当前页
     */
    private static final long DEFAULT_PAGE_NUMBER = 1L;

    /**
     * 默认每页的数量
     */
    private static final long DEFAULT_PAGE_SIZE = 50L;

    /**
     * 默认导航页号数量
     */
    private static final long DEFAULT_NAVIGATE_PAGES = 10L;

    /**
     * 获取分页信息<br/>
     * 当前页小于1时,当前页为1<br/>
     * 当前页大于总页数时,当前页为总页数
     *
     * @param pageNum 当前页
     * @param pageSize 每页的数量
     * @param count 总记录数
     * @param list 结果集
     * @param navigatePages 导航页码数
     * @return {@link PageInfo} 分页信息
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static <T> PageInfo<T> getPageInfo(long pageNum, long pageSize, long count, List<T> list, long navigatePages) {
        PageInfo<T> pageInfo = new PageInfo<>();

        // 结果集(如果为null则填充一个空的List集合)
        List<T> validList = getList(list);
        pageInfo.setList(validList);

        // 每页的数量(小于1则等于1)
        long validPageSize = getPageSize(pageSize);
        pageInfo.setPageSize(validPageSize);

        // 当前页的数量(结果集的大小)
        int size = validList.size();
        pageInfo.setSize(size);

        // 总记录数(执行count(*)语句返回的结果)
        long total = count;
        pageInfo.setTotal(total);

        // 总页数(总记录数 / 每页的数量,如果 总记录数 % 每页的数量 != 0,总页数+1 )
        long pages = getPages(total, validPageSize);
        pageInfo.setPages(pages);

        // 当前页(小于1则等于1,大于总页数则等于总页数)
        long validPageNum = getPageNum(pageNum, pages);
        pageInfo.setPageNum(validPageNum);

        // 第一页的页码(第一页的常量)
        long firstPage = DEFAULT_PAGE_NUMBER;
        pageInfo.setFirstPage(firstPage);

        // 最后一页的页码(总页数)
        long lastPage = pages;
        pageInfo.setLastPage(lastPage);

        // 上一页的页码(当前页-1)
        long prevPage = validPageNum - DEFAULT_PAGE_NUMBER;
        pageInfo.setPrevPage(prevPage);

        // 下一页的页码(当前页+1)
        long nextPage = getNextPage(validPageNum, lastPage);
        pageInfo.setNextPage(nextPage);

        // 是否为第一页(当前页 == 第一页的页码)
        boolean hasFirstPage = validPageNum == firstPage;
        pageInfo.setHasFirstPage(hasFirstPage);

        // 是否为最后一页(当前页 == 最后一页的页码)
        boolean hasLastPage = validPageNum == lastPage;
        pageInfo.setHasLastPage(hasLastPage);

        // 是否有上一页(!是否为第一页)
        boolean hasPreviousPage = !hasFirstPage;
        pageInfo.setHasPreviousPage(hasPreviousPage);

        // 是否有下一页(!是否为最后一页)
        boolean hasNextPage = !hasLastPage;
        pageInfo.setHasNextPage(hasNextPage);

        // 导航页码数
        long validNavigatePages = navigatePages;
        pageInfo.setNavigatePages(validNavigatePages);

        // 所有导航页号
        long[] navigatePageNums = getNavigatePageNums(validNavigatePages, pages, validPageNum, lastPage);
        pageInfo.setNavigatePageNums(navigatePageNums);

        // 导航页号第一页的页码(所有导航页号中的第一个)
        long navigateFirstPage = navigatePageNums[0];
        pageInfo.setNavigateFirstPage(navigateFirstPage);

        // 导航页号最后一页的页码(所有导航页号中的最后一个)
        long navigateLastPage = navigatePageNums[navigatePageNums.length - 1];
        pageInfo.setNavigateLastPage(navigateLastPage);

        return pageInfo;
    }

    /**
     * 获取分页信息(使用默认的每页数量与导航页码数)<br/>
     * 当前页小于1时,当前页为1<br/>
     * 当前页大于总页数时,当前页为总页数
     *
     * @param pageNum 当前页
     * @param count 总记录数
     * @param list 结果集
     * @return {@link PageInfo} 分页信息
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static <T> PageInfo<T> getPageInfo(long pageNum, long count, List<T> list) {
        return getPageInfo(pageNum, DEFAULT_PAGE_SIZE, count, list, DEFAULT_NAVIGATE_PAGES);
    }

    /**
     * 获取空集合的分页信息(使用默认的每页数量与导航页码数)<br/>
     * 且总记录数为0,结果集为null
     *
     * @param pageNum 当前页
     * @return {@link PageInfo} 分页信息
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static <T> PageInfo<T> getPageInfoForEmptyList(long pageNum) {
        return getPageInfo(pageNum, DEFAULT_PAGE_SIZE, 0, null, DEFAULT_NAVIGATE_PAGES);
    }

    /**
     * 获取分页查询对象(使用默认的每页数量)<br/>
     * 当前页小于1时,当前页为1
     *
     * @param pageNum 当前页
     * @return {@link PageQuery}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static PageQuery getPageQueryByDefaultPageSize(long pageNum) {
        return getPageQuery(pageNum, DEFAULT_PAGE_SIZE);
    }

    /**
     * 获取分页查询对象(使用默认的每页数量)<br/>
     * 当前页小于1时,当前页为1<br/>
     * 当前页大于总页数时,当前页为总页数
     *
     * @param pageNum 当前页
     * @param count 总记录数
     * @return {@link PageQuery}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static PageQuery getPageQueryByDefaultPageSize(long pageNum, long count) {
        return getPageQuery(pageNum, DEFAULT_PAGE_SIZE, count);
    }

    /**
     * 获取分页查询对象<br/>
     * 当前页小于1时,当前页为1
     *
     * @param pageNum 当前页
     * @param pageSize 每页的数量
     * @return {@link PageQuery}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static PageQuery getPageQuery(long pageNum, long pageSize) {
        PageQuery query = new PageQuery();
        query.setOffset(getOffset(pageNum, pageSize));
        query.setSize(pageSize);
        return query;
    }

    /**
     * 获取分页查询对象<br/>
     * 当前页小于1时,当前页为1<br/>
     * 当前页大于总页数时,当前页为总页数
     *
     * @param pageNum 当前页
     * @param pageSize 每页的数量
     * @param count 总记录数
     * @return {@link PageQuery}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    public static PageQuery getPageQuery(long pageNum, long pageSize, long count) {
        PageQuery query = new PageQuery();
        query.setOffset(getOffset(pageNum, pageSize, count));
        query.setSize(pageSize);
        return query;
    }

    /**
     * 获取偏移量
     *
     * @param pageNum 当前页
     * @param pageSize 每页的数量
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getOffset(long pageNum, long pageSize) {
        return (getPageNum(pageNum) - 1) * pageSize;
    }

    /**
     * 获取偏移量
     *
     * @param pageNum 当前页
     * @param pageSize 每页的数量
     * @param count 总记录数
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getOffset(long pageNum, long pageSize, long count) {
        pageNum = getPageNum(pageNum);
        // 总页数
        long pages = getPages(count, pageSize);

        // 如果请求的页码大于总页数的话
        boolean isInvalid = pageNum > pages;
        if (isInvalid) {
            pageNum = pages;
        }
        return (pageNum - 1) * pageSize;
    }

    /**
     * 获取当前页
     *
     * @param pageNum 当前页
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getPageNum(long pageNum) {
        boolean isInvalid = pageNum < DEFAULT_PAGE_NUMBER;
        if (isInvalid) {
            pageNum = DEFAULT_PAGE_NUMBER;
        }
        return pageNum;
    }

    /**
     * 获取结果集
     *
     * @param list 结果集
     * @return {@link java.util.List<T>}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static <T> List<T> getList(List<T> list) {
        // 结果集(如果为null则填充一个空的List集合)
        return Optional.ofNullable(list).orElse(new ArrayList<>(0));
    }

    /**
     * 获取每页的数量
     *
     * @param pageSize 每页的数量
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getPageSize(long pageSize) {
        // 每页的数量(小于1则等于1)
        long validPageSize = pageSize;
        if (validPageSize < DEFAULT_PAGE_NUMBER) {
            validPageSize = DEFAULT_PAGE_NUMBER;
        }
        return validPageSize;
    }

    /**
     * 获取总页数
     *
     * @param total 总记录数
     * @param validPageSize 有效的每页数量
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getPages(long total, long validPageSize) {
        // 总页数(总记录数 / 每页的数量,如果 总记录数 % 每页的数量 != 0,总页数+1 )
        long pages = total / validPageSize;
        if (total % validPageSize != 0) {
            pages++;
        }
        if (pages < DEFAULT_PAGE_NUMBER) {
            pages = DEFAULT_PAGE_NUMBER;
        }
        return pages;
    }

    /**
     * 获取当前页
     *
     * @param pageNum 当前页
     * @param pages 总页数
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getPageNum(long pageNum, long pages) {
        // 当前页(小于1则等于1,大于总页数则等于总页数)
        long validPageNum = pageNum;
        if (validPageNum < DEFAULT_PAGE_NUMBER) {
            validPageNum = DEFAULT_PAGE_NUMBER;
        } else if (validPageNum > pages) {
            validPageNum = pages;
        }
        return validPageNum;
    }

    /**
     * 获取下一页
     *
     * @param validPageNum 有效的当前页
     * @param lastPage 最后一页
     * @return {@link long}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long getNextPage(long validPageNum, long lastPage) {
        // 下一页的页码(当前页+1)
        long nextPage = validPageNum + DEFAULT_PAGE_NUMBER;
        if (nextPage > lastPage) {
            nextPage = DEFAULT_PAGE_NUMBER - 1;
        }
        return nextPage;
    }

    /**
     * 获取导航页号
     *
     * @param validNavigatePages 有效的导航页码数
     * @param pages 总页数
     * @param validPageNum 有效的当前页
     * @param lastPage 最后一页
     * @return {@link long[]}
     * @author yangqiaoxin
     * @date 2021/01/01
     */
    private static long[] getNavigatePageNums(long validNavigatePages,
                                              long pages,
                                              long validPageNum,
                                              long lastPage) {
        // 所有导航页号
        long start;
        long mid = validNavigatePages >> 1; // 中间索引 = 导航页码数 / 2
        long end;
        if (pages <= validNavigatePages) { // 总页数 <= 导航页码数
            start = DEFAULT_PAGE_NUMBER; // 起始 = 1
            end = pages; // 结束 = 总页数
        } else if (validPageNum <= mid + 1) { // 当前页 <= 最左范围的中间页码(即:中间索引 + 1)
            start = DEFAULT_PAGE_NUMBER; // 起始 = 1
            end = validNavigatePages; // 结束 = 导航页码数
        } else if (validPageNum >= lastPage - mid + 1) { // 当前页 >= 最右范围的中间页码(即:最后一页的页码 - 中间索引 + 1)
            end = lastPage; // 结束 = 最后一页的页码
            start = end - validNavigatePages + 1; // 起始 = 结束 - 导航页码数 + 1
        } else { // 否则
            start = validPageNum - mid; // 起始 = 当前页 - 中间索引
            end = start + validNavigatePages - 1; // 结束 = 起始 + 导航页码数 - 1
        }
        int capacity = (int) (end - start + 1);
        long[] navigatePageNums;
        if (capacity > 0) {
            navigatePageNums = new long[capacity];
            int i = 0;
            for (long val = start; val <= end; val++) {
                navigatePageNums[i++] = val;
            }
        } else {
            navigatePageNums = new long[(int) DEFAULT_PAGE_NUMBER];
            navigatePageNums[0] = DEFAULT_PAGE_NUMBER;
        }
        return navigatePageNums;
    }

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容