Java中两种分页遍历的使用姿势

简介: 在日常开发中,分页遍历迭代的场景可以说非常普遍了,比如扫表,每次捞100条数据,然后遍历这100条数据,依次执行某个业务逻辑;这100条执行完毕之后,再加载下一百条数据,直到扫描完毕那么要实现上面这种分页迭代遍历的场景,我们可以怎么做呢

在日常开发中,分页遍历迭代的场景可以说非常普遍了,比如扫表,每次捞100条数据,然后遍历这100条数据,依次执行某个业务逻辑;这100条执行完毕之后,再加载下一百条数据,直到扫描完毕


那么要实现上面这种分页迭代遍历的场景,我们可以怎么做呢


本文将介绍两种使用姿势


  • 常规的使用方法
  • 借助Iterator的使用姿势


1. 数据查询模拟



首先mock一个分页获取数据的逻辑,直接随机生成数据,并且控制最多返回三页


public static int cnt = 0;
private static List<String> randStr(int start, int size) {
    ++cnt;
    if (cnt > 3) {
        return Collections.emptyList();
    } else if (cnt == 3) {
        cnt = 0;
        size -= 2;
    }
    System.out.println("======================= start to gen randList ====================");
    List<String> ans = new ArrayList<>(size);
    for (int i = 0; i < size; i++) {
        ans.add((start + i) + "_" + UUID.randomUUID().toString());
    }
    return ans;
}
复制代码


2. 基本实现方式



针对这种场景,最常见也是最简单直观的实现方式


  • while死循环
  • 内部遍历
private static void scanByNormal() {
    int start = 0;
    int size = 5;
    while (true) {
        List<String> list = randStr(start, size);
        for (String str : list) {
            System.out.println(str);
        }
        if (list.size() < size) {
            break;
        }
        start += list.size();
    }
}
复制代码


3. 迭代器实现方式



接下来介绍一种更有意思的方式,借助迭代器的遍历特性来实现,首先自定义一个通用分页迭代器


public static abstract class MyIterator<T> implements Iterator<T> {
    private int start = 0;
    private int size = 5;
    private int currentIndex;
    private boolean hasMore = true;
    private List<T> list;
    public MyIterator() {
    }
    @Override
    public boolean hasNext() {
        if (list != null && list.size() > currentIndex) {
            return true;
        }
        // 当前的数据已经加载完毕,尝试加载下一批
        if (!hasMore) {
            return false;
        }
        list = load(start, size);
        if (list == null || list.isEmpty()) {
            // 没有加载到数据,结束
            return false;
        }
        if (list.size() < size) {
            // 返回条数小于限制条数,表示还有更多的数据可以加载
            hasMore = false;
        }
        currentIndex = 0;
        start += list.size();
        return true;
    }
    @Override
    public T next() {
        return list.get(currentIndex++);
    }
    public abstract List<T> load(int start, int size);
}
复制代码


接下来借助上面的迭代器可以比较简单的实现我们的需求了


private static void scanByIterator() {
    MyIterator<String> iterator = new MyIterator<String>() {
        @Override
        public List<String> load(int start, int size) {
            return randStr(start, size);
        }
    };
    while (iterator.hasNext()) {
        String str = iterator.next();
        System.out.println(str);
    }
}
复制代码


那么问题来了,上面这种使用方式比前面的优势体现再哪儿呢?


  • 双层循环改为单层循环


接下来接入重点了,在jdk1.8引入了函数方法 + lambda之后,又提供了一个更简洁的使用姿势


public class IteratorTestForJdk18 {
    @FunctionalInterface
    public interface LoadFunc<T> {
        List<T> load(int start, int size);
    }
    public static class MyIterator<T> implements Iterator<T> {
        private int start = 0;
        private int size = 5;
        private int currentIndex;
        private boolean hasMore = true;
        private List<T> list;
        private LoadFunc<T> loadFunc;
        public MyIterator(LoadFunc<T> loadFunc) {
            this.loadFunc = loadFunc;
        }
        @Override
        public boolean hasNext() {
            if (list != null && list.size() > currentIndex) {
                return true;
            }
            // 当前的数据已经加载完毕,尝试加载下一批
            if (!hasMore) {
                return false;
            }
            list = loadFunc.load(start, size);
            if (list == null || list.isEmpty()) {
                // 没有加载到数据,结束
                return false;
            }
            if (list.size() < size) {
                // 返回条数小于限制条数,表示还有更多的数据可以加载
                hasMore = false;
            }
            currentIndex = 0;
            start += list.size();
            return true;
        }
        @Override
        public T next() {
            return list.get(currentIndex++);
        }
    }
}
复制代码


在jdk1.8及之后的使用姿势,一行代码即可


private static void scanByIteratorInJdk8() {
    new MyIterator<>(IteratorTestForJdk18::randStr)
        .forEachRemaining(System.out::println);
}
复制代码


这次对比效果是不是非常显眼了,从此以后分页迭代遍历再也不用冗长的双重迭代了



相关文章
|
6月前
|
存储 Java
Java学习笔记 List集合的定义、集合的遍历、迭代器的使用
Java学习笔记 List集合的定义、集合的遍历、迭代器的使用
|
8月前
|
存储 算法 Java
Java中,树与图的算法涉及二叉树的前序、中序、后序遍历以及DFS和BFS搜索。
【6月更文挑战第21天】Java中,树与图的算法涉及二叉树的前序、中序、后序遍历以及DFS和BFS搜索。二叉树遍历通过访问根、左、右子节点实现。DFS采用递归遍历图的节点,而BFS利用队列按层次访问。以下是简化的代码片段:[Java代码略]
57 4
|
3月前
|
存储 Java 开发者
在 Java 中,如何遍历一个 Set 集合?
【10月更文挑战第30天】开发者可以根据具体的需求和代码风格选择合适的遍历方式。增强for循环简洁直观,适用于大多数简单的遍历场景;迭代器则更加灵活,可在遍历过程中进行更多复杂的操作;而Lambda表达式和`forEach`方法则提供了一种更简洁的函数式编程风格的遍历方式。
|
4月前
|
Java 程序员 编译器
Java|如何正确地在遍历 List 时删除元素
从源码分析如何正确地在遍历 List 时删除元素。为什么有的写法会导致异常,而另一些不会。
80 3
|
4月前
|
Java 程序员 调度
Java|PageHelper 怎么自作主张帮我分页?
没有调用 PageHelper.startPage,查询怎么也被自动分页了?
20 2
|
4月前
|
前端开发 小程序 Java
java基础:map遍历使用;java使用 Patten 和Matches 进行正则匹配;后端传到前端展示图片三种情况,并保存到手机
这篇文章介绍了Java中Map的遍历方法、使用Pattern和matches进行正则表达式匹配,以及后端向前端传输图片并保存到手机的三种情况。
45 1
|
4月前
|
存储 算法 Java
Java一分钟之-数组的创建与遍历
数组作为Java中存储和操作一组相同类型数据的基本结构,其创建和遍历是编程基础中的基础。通过不同的创建方式,可以根据实际需求灵活地初始化数组。而选择合适的遍历方法,则可以提高代码的可读性和效率。掌握这些基本技能,对于深入学习Java乃至其他编程语言的数据结构和算法都是至关重要的。
36 6
|
5月前
|
域名解析 分布式计算 网络协议
java遍历hdfs路径信息,报错EOFException
java遍历hdfs路径信息,报错EOFException
51 3
|
6月前
|
Java 容器
07 Java数组与数组操作(定义+遍历+排序+增删改查)(上)
07 Java数组与数组操作(定义+遍历+排序+增删改查)
73 8
|
6月前
|
存储 Java API
07 Java数组与数组操作(定义+遍历+排序+增删改查)(下)
07 Java数组与数组操作(定义+遍历+排序+增删改查)
50 4