当前位置: 首页 > news >正文

设计模式:迭代器模式

迭代器模式的示例可以涵盖各种数据结构的遍历,包括数组、列表、树、图等。下面是一些不同场景下迭代器模式的示例及其代码实现。

示例 1: 数组遍历

使用迭代器模式遍历数组。

// 迭代器接口
interface Iterator<T> {boolean hasNext();T next();
}// 数组迭代器
class ArrayIterator<T> implements Iterator<T> {private T[] array;private int currentIndex = 0;public ArrayIterator(T[] array) {this.array = array;}@Overridepublic boolean hasNext() {return currentIndex < array.length;}@Overridepublic T next() {if (!hasNext()) {throw new NoSuchElementException();}return array[currentIndex++];}
}// 客户端代码
public class ArrayIteratorExample {public static void main(String[] args) {Integer[] numbers = {1, 2, 3, 4, 5};Iterator<Integer> iterator = new ArrayIterator<>(numbers);while (iterator.hasNext()) {Integer number = iterator.next();System.out.println(number);}}
}

示例 2: 二叉树的深度优先遍历

二叉树的深度优先遍历(使用栈实现)。

// 树节点
class TreeNode<T> {T value;TreeNode<T> left;TreeNode<T> right;TreeNode(T value) {this.value = value;}
}// 树的迭代器
class BinaryTreeIterator<T> implements Iterator<T> {private Stack<TreeNode<T>> stack = new Stack<>();public BinaryTreeIterator(TreeNode<T> root) {pushLeft(root);}private void pushLeft(TreeNode<T> node) {while (node != null) {stack.push(node);node = node.left;}}@Overridepublic boolean hasNext() {return !stack.isEmpty();}@Overridepublic T next() {if (!hasNext()) {throw new NoSuchElementException();}TreeNode<T> node = stack.pop();pushLeft(node.right);return node.value;}
}// 客户端代码
public class BinaryTreeIteratorExample {public static void main(String[] args) {TreeNode<Integer> root = new TreeNode<>(1);root.left = new TreeNode<>(2);root.right = new TreeNode<>(3);root.left.left = new TreeNode<>(4);root.left.right = new TreeNode<>(5);Iterator<Integer> iterator = new BinaryTreeIterator<>(root);while (iterator.hasNext()) {Integer value = iterator.next();System.out.println(value);}}
}

示例 3: 分页遍历

在分页查询场景中使用迭代器模式。

// 分页迭代器
class PagingIterator<T> implements Iterator<T> {private Iterator<T> currentIterator;private int pageSize;private int currentPage = 0;private Supplier<List<T>> pageSupplier;public PagingIterator(int pageSize, Supplier<List<T>> pageSupplier) {this.pageSize = pageSize;this.pageSupplier = pageSupplier;fetchNextPage();}private void fetchNextPage() {List<T> page = pageSupplier.get();currentIterator = page.iterator();currentPage++;}@Overridepublic boolean hasNext() {if (currentIterator.hasNext()) {return true;}fetchNextPage(); // 尝试获取下一页return currentIterator.hasNext();}@Overridepublic T next() {if (!hasNext()) {throw new NoSuchElementException();}return currentIterator.next();}
}// 模拟分页数据提供函数
Supplier<List<Integer>> pageSupplier = new Supplier<List<Integer>>() {private int totalPages = 5;private int currentPage = 0;@Overridepublic List<Integer> get() {if (currentPage < totalPages) {currentPage++;return Arrays.asList(1, 2, 3); // 模拟每页的数据}return Collections.emptyList();}
};// 客户端代码
public class PagingIteratorExample {public static void main(String[] args) {Iterator<Integer> iterator = new PagingIterator<>(3, pageSupplier);while (iterator.hasNext()) {Integer number = iterator.next();System.out.println(number);}}
}

总结与建议

以上示例展示了迭代器模式在不同数据结构遍历上的应用。迭代器模式的关键优势是它提供了一种统一的接口来遍历各种类型的数据结构,同时对客户端隐藏了数据结构的实现细节。

在实现迭代器模式时,建议:

  • 保持迭代器接口简单,通常包含 hasNext()next() 方法即可。
  • 确保迭代器正确处理底层数据结构的变更。
  • 考虑迭代器的线程安全性,特别是在多线程环境中使用共享数据结构时。
  • 如果迭代逻辑非常复杂,可以考虑使用访问者模式来进一步分离逻辑和数据结构。

迭代器模式是一种强大的工具,可以使代码更加清晰、灵活,并且易于维护。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • [StartingPoint][Tier1]Pennyworth
  • 【LeetCode热题100】33. 搜索旋转排序数组(二分)
  • Java后端开发中Java 8,JVM和JDK的关系
  • C语言如何声明外部变量?
  • 一条SQL查询语句的执行顺序
  • mysql慢sql排查与分析
  • Blender怎么样启动默认移动和Cavity效果
  • 理解 Golang 变量在内存分配中的规则
  • ics-05-攻防世界
  • 爬取高校专业信息的Python爬虫简介与实践
  • 【C++ STL算法】sort 排序
  • 隐私计算实训营学习七:隐语SCQL的架构详细拆解
  • 数据库的基本操作
  • 面试题多态结合线程
  • 【TI毫米波雷达】IWR6843AOP的官方文件资源名称BUG,选择xwr68xx还是xwr64xx,及需要注意的问题
  • 自己简单写的 事件订阅机制
  • 002-读书笔记-JavaScript高级程序设计 在HTML中使用JavaScript
  • Apache的基本使用
  • C++类中的特殊成员函数
  • canvas实际项目操作,包含:线条,圆形,扇形,图片绘制,图片圆角遮罩,矩形,弧形文字...
  • Docker 1.12实践:Docker Service、Stack与分布式应用捆绑包
  • Git学习与使用心得(1)—— 初始化
  • go语言学习初探(一)
  • iOS小技巧之UIImagePickerController实现头像选择
  • Javascript编码规范
  • Java比较器对数组,集合排序
  • Python学习之路13-记分
  • quasar-framework cnodejs社区
  • storm drpc实例
  • vue2.0开发聊天程序(四) 完整体验一次Vue开发(下)
  • XForms - 更强大的Form
  • 短视频宝贝=慢?阿里巴巴工程师这样秒开短视频
  • 给第三方使用接口的 URL 签名实现
  • 前端学习笔记之原型——一张图说明`prototype`和`__proto__`的区别
  • 什么软件可以提取视频中的音频制作成手机铃声
  • 移动端 h5开发相关内容总结(三)
  • 优秀架构师必须掌握的架构思维
  • ​软考-高级-信息系统项目管理师教程 第四版【第19章-配置与变更管理-思维导图】​
  • ###51单片机学习(2)-----如何通过C语言运用延时函数设计LED流水灯
  • #includecmath
  • (007)XHTML文档之标题——h1~h6
  • (1)Map集合 (2)异常机制 (3)File类 (4)I/O流
  • (21)起落架/可伸缩相机支架
  • (第三期)书生大模型实战营——InternVL(冷笑话大师)部署微调实践
  • (一)80c52学习之旅-起始篇
  • (转)创业家杂志:UCWEB天使第一步
  • .bat批处理(六):替换字符串中匹配的子串
  • .NET Framework、.NET Core 、 .NET 5、.NET 6和.NET 7 和.NET8 简介及区别
  • .NET 的程序集加载上下文
  • .NET 设计模式初探
  • .NET/C# 的字符串暂存池
  • .NET关于 跳过SSL中遇到的问题
  • @Autowired @Resource @Qualifier的区别
  • @EnableAsync和@Async开始异步任务支持
  • @param注解什么意思_9000字,通俗易懂的讲解下Java注解