• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

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

java 搞代码 4年前 (2022-01-05) 35次浏览 已收录 0个评论

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

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

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

本文将介绍两种使用姿势

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

1. 数据查询模拟

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

 public static int cnt = 0; private static List 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 ans = new ArrayList(size); for (int i = 0; i <size; <strong style="color:transparent">来源gaodai#ma#com搞@@代~&码网</strong>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 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 implements Iterator { private int start = 0; private int size = 5; private int currentIndex; private boolean hasMore = true; private List 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 load(int start, int size); } 

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

 private static void scanByIterator() { MyIterator iterator = new MyIterator() { @Override public List 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 { List load(int start, int size); } public static class MyIterator implements Iterator { private int start = 0; private int size = 5; private int currentIndex; private boolean hasMore = true; private List list; private LoadFunc loadFunc; public MyIterator(LoadFunc 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); } 

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

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

以上就是详解Java中两种分页遍历的使用姿势的详细内容,更多请关注gaodaima搞代码网其它相关文章!


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:详解Java中两种分页遍历的使用姿势

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址