赞
踩
迭代器模式定义如下:提供一种方法访问一个容器对象中的各个元素,而又勿需暴露该对象的内部细节,是一种只应用于容器对象遍历的设计模式。
适合迭代器模式的情景如下:
- 遍历集合对象,不需要知道对象在集合中的存储方式
- 用户可以同时使用多个迭代器遍历一个集合
在计算机程序设计中,经常会用到各种各样的容器,如数组、链表、集合、映射等。在容器的各种操作中,元素遍历时最常见的操作之一。不同类型的容器,遍历方法也不同。以学生数组和链表元素遍历功能为例,代码如下:
(1)学生类
- public class Student {
- String sno;
- String name;
-
- public Student(String sno, String name) {
- this.sno = sno;
- this.name = name;
- }
- }
(2)学生数组容器类
该类是泛型数组容器类,提供遍历功能的基础方法是:获得数组大小size()方法,获得数组某下标位置元素对象get()方法。
- public class MyArray<T> {
- T t[];
- //传入一个数组
- MyArray(T t[]) {
- this.t = t;
- }
- T get(int n) {
- if (n < t.length) {
- return t[n];
- } else {
- return null;
- }
- }
- int size() {
- return t.length;
- }
- }
(3)链表节点类
该类是任意泛型T的链表节点类,成员变量next代表指向的下一个Node节点的地址。
- public class Node<T> {
- T t;
- Node next;
- Node(T t) {
- this.t = t;
- }
- void setNext(Node next) {
- this.next = next;
- }
- Node getNext() {
- return next;
- }
- T getT() {
- return t;
- }
- }
(4)链表容器类
对链表容器来说,最重要的就是首元素,因此将其定义为成员变量head,getNext()方法是遍历链表最重要的基础方法,功能是返回形参节点node的后继节点。
- public class MyLink<T> {
- Node<T> head;
- MyLink(Node<T> head) {
- this.head = head;
- }
-
- public Node<T> getHead() {
- return head;
- }
- Node<T> getNext(Node<T> node) {
- return node.getNext();
- }
- }
(5)遍历数组及链表的测试类
- public class Test {
- public static void main(String[] args) {
- System.out.println("数组测试:");
- //初始化一个学生数组
- Student s[] = new Student[2];
- s[0] = new Student("1001","zhangyin");
- s[1] = new Student("1002","siminmin");
- //将该学生数组传入MyArray
- MyArray<Student> myArray = new MyArray<>(s);
- for (int i = 0; i < myArray.size(); i++) {
- //返回数组下标对应的学生元素
- Student student = myArray.get(i);
- System.out.println(student.sno+"\t"+student.name);
- }
- System.out.println("链表测试:");
- //初始化另一个学生数组
- Student s1[] = new Student[3];
- s1[0] = new Student("1003","yinyin");
- s1[1] = new Student("1004","minmin");
- s1[2] = new Student("1005","sisi");
- //将每个学生元素放入node中
- Node<Student> node1 = new Node<>(s1[0]);
- Node<Student> node2 = new Node<>(s1[1]);
- Node<Student> node3 = new Node<>(s1[2]);
- node1.next = node2;
- node2.next = node3;
-
- MyLink<Student> myLink = new MyLink<>(node1);
- Node<Student> cur = myLink.getHead();
- while (cur != null) {
- Student student = cur.getT();
- System.out.println(student.sno+"\t"+student.name);
- cur = cur.getNext();
- }
-
- }
- }
测试结果:
- 数组测试:
- 1001 zhangyin
- 1002 siminmin
- 链表测试:
- 1003 yinyin
- 1004 minmin
- 1005 sisi
思考:
从上面例子可以看出,数组和链表的遍历方式是不同的,那么能否有一种容器元素遍历方式,对各种容器而言都是一致的,且与容器的性质无关呢?这就用到迭代器遍历的思想。
迭代器模式,提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部元素的表示。迭代器模式把在元素之间游走的责任交给了迭代器,而不是聚合对象。
JDK中迭代器模式是很成熟的,它应用在许多容器中,下面代码体现了迭代器在Vector、LinkedList、Set容器中的遍历应用,从中可以更清晰地看出迭代器的特点。
Vector容器的特点是内存空间是线性、连续的;
LinkedList容器的特点是链式结构;
Set容器的特点是树形结构
通过下面代码可以知道,容器元素的遍历与具体的容器是无关的。
- public class Test2 {
-
- public static void traverse(Collection<String> collection) {
- Iterator<String> it = collection.iterator();
- while (it.hasNext()) {
- String str = it.next();
- System.out.println(str);
- }
- }
-
- public static void main(String[] args) {
- Vector<String> vector = new Vector<>();
- vector.add("aaa");
- vector.add("bbb");
- vector.add("ccc");
- System.out.println("Vector traverse:");
- traverse(vector);
- System.out.println();
-
- LinkedList<String> linkedList = new LinkedList<>();
- linkedList.add("ddd");
- linkedList.add("eee");
- linkedList.add("fff");
- System.out.println("LinkedList traverse:");
- traverse(linkedList);
- System.out.println();
-
- Set<String> set = new HashSet<>();
- set.add("ggg");
- set.add("hhh");
- set.add("iii");
- System.out.println("Set traverse:");
- traverse(set);
- System.out.println();
- }
- }
测试结果:
- Vector traverse:
- aaa
- bbb
- ccc
-
- LinkedList traverse:
- ddd
- eee
- fff
-
- Set traverse:
- ggg
- iii
- hhh
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。