服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|JavaScript|易语言|

服务器之家 - 编程语言 - Java教程 - 详解Java中两种分页遍历的使用姿势

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

2021-08-18 12:04一灰灰 Java教程

这篇文章主要介绍了详解Java中两种分页遍历的使用姿势,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

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

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

本文将介绍两种使用姿势

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

1. 数据查询模拟

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
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死循环
  • 内部遍历
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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. 迭代器实现方式

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
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);
}

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
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之后,又提供了一个更简洁的使用姿势

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
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及之后的使用姿势,一行代码即可

?
1
2
3
4
private static void scanByIteratorInJdk8() {
  new MyIterator<>(IteratorTestForJdk18::randStr)
    .forEachRemaining(System.out::println);
}

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

到此这篇关于详解Java中两种分页遍历的使用姿势的文章就介绍到这了,更多相关Java 分页遍历内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家! 

原文链接:https://juejin.cn/post/6934913224689057799

延伸 · 阅读

精彩推荐
  • Java教程java数据类型与二进制详细介绍

    java数据类型与二进制详细介绍

    这篇文章主要介绍了java数据类型与二进制详细介绍的相关资料,这里对数据类型进行了一一介绍分析,并说明自动转换和强制转换,需要的朋友可以参考下...

    1473rg3372020-12-05
  • Java教程intellij IDEA配置springboot的图文教程

    intellij IDEA配置springboot的图文教程

    Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。接下来通过本文给大家介绍intellij IDEA配置springboot的...

    chen_NeverRetreat12152021-04-12
  • Java教程java编程基础之模仿用户登录代码分享

    java编程基础之模仿用户登录代码分享

    这篇文章主要介绍了java编程基础之模仿用户登录代码分享,小编觉得挺不错的,这里分享给大家,供需要的朋友参考。...

    忘了长发模样5712021-01-23
  • Java教程简单验证码生成Java版

    简单验证码生成Java版

    这篇文章主要为大家详细介绍了简单验证码生成Java版,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    刘水镜12062021-02-20
  • Java教程Java实现洗牌发牌的方法

    Java实现洗牌发牌的方法

    这篇文章主要介绍了Java实现洗牌发牌的方法,涉及java针对数组的遍历与排序操作相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下 ...

    罪恶的花生4052019-12-29
  • Java教程java基础之Collection与Collections和Array与Arrays的区别

    java基础之Collection与Collections和Array与Arrays的区别

    这篇文章主要介绍了java基础之Collection与Collections和Array与Arrays的区别的相关资料,本文主要说明两者的区别以防大家混淆概念,需要的朋友可以参考下...

    lucherr1612020-12-18
  • Java教程IDEA内存调试插件(好用)

    IDEA内存调试插件(好用)

    本文给大家分享IDEA中一个很有用的内存调试插件,非常不错,具有参考借鉴价值,需要的朋友参考下...

    azhegps9142021-04-07
  • Java教程Java封装、继承、多态三大特征的理解

    Java封装、继承、多态三大特征的理解

    封装、继承、多态三大特征是java中比较常用的,务必要掌握,下面给大家介绍Java封装、继承、多态三大特征的理解,有不清楚的朋友可以一起学习下 ...

    hansheng5202020-05-18