JAVA设计模式(16) —<行为型>迭代子模式(Iterator)

本文深入探讨了迭代器模式的定义、结构、代码实现、优缺点、应用场景、注意事项及扩展,通过具体代码实例展示了如何在Java中使用迭代器模式进行数据遍历,并通过重构示例说明了迭代器模式在解决实际问题时的优势。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

定义:

迭代子模式(Iterator)

Provide a way to access the elements of an aggregate object sequentially without exposing its underlying representation.(它提供一种方法访问一个容器对象中各个元素,而又不暴露该对象的内部细节。)

此模式是一个没落的模式,基本上没人会单独写一个迭代器模式。没落的原因是,绝大多数需要迭代器的场合,都已经有了标准的实现。

1.1 通用类图:

在迭代器模式结构图中包含如下几个角色:

       ● 抽象迭代器(Iterator):它定义了访问和遍历元素的接口,声明了用于遍历数据元素的方法,例如:用于获取第一个元素的first()方法,用于访问下一个元素的next()方法,用于判断是否还有下一个元素的hasNext()方法,用于获取当前元素的currentItem()方法等,在具体迭代器中将实现这些方法。

       ● 具体迭代器(ConcreteIterator):它实现了抽象迭代器接口,完成对聚合对象的遍历,同时在具体迭代器中通过游标来记录在聚合对象中所处的当前位置,在具体实现时,游标通常是一个表示位置的非负整数。

       ● 抽象聚合类(Aggregate):它用于存储和管理元素对象,声明一个createIterator()方法用于创建一个迭代器对象,充当抽象迭代器工厂角色。一般是一个接口,提供一个iterator()方法,例如java中的Collection接口,List接口,Set接口等

       ● 具体聚合类(ConcreteAggregate)它实现了在抽象聚合类中声明的createIterator()方法,该方法返回一个与该具体聚合类对应的具体迭代器ConcreteIterator实例。比如List接口的有序列表实现ArrayList,List接口的链表实现LinkList,Set接口的哈希列表的实现HashSet等。

       在迭代器模式中,提供了一个外部的迭代器来对聚合对象进行访问和遍历,迭代器定义了一个访问该聚合元素的接口,并且可以跟踪当前遍历的元素,了解哪些元素已经遍历过而哪些没有。迭代器的引入,将使得对一个复杂聚合对象的操作变得简单。

       下面我们结合代码来对迭代器模式的结构进行进一步分析。在迭代器模式中应用了工厂方法模式,抽象迭代器对应于抽象产品角色,具体迭代器对应于具体产品角色,抽象聚合类对应于抽象工厂角色具体聚合类对应于具体工厂角色


1.2 通用代码:

[java]  view plain copy
  1. public interface Aggregate {  
  2.     // 是容器必然有元素的增加  
  3.     public void add(Object object);  
  4.   
  5.     // 减少元素  
  6.     public void remove(Object object);  
  7.   
  8.     // 由迭代器来遍历所有的元素  
  9.     public Iterator iterator();  
  10. }  
  11.   
  12. public class ConcreteAggregate implements Aggregate {  
  13.     // 容纳对象的容器  
  14.     private Vector vector = new Vector();  
  15.   
  16.     // 增加一个元素  
  17.     public void add(Object object) {  
  18.         this.vector.add(object);  
  19.     }  
  20.   
  21.     // 返回迭代器对象  
  22.     public Iterator iterator() {  
  23.         return new ConcreteIterator(this.vector);  
  24.     }  
  25.   
  26.     // 删除一个元素  
  27.     public void remove(Object object) {  
  28.         this.vector.remove(object);  
  29.     }  
  30. }  
  31.   
  32. public interface Iterator {  
  33.     // 遍历到下一个元素  
  34.     public Object next();  
  35.   
  36.     // 是否已经遍历到尾部  
  37.     public boolean hasNext();  
  38.   
  39.     // 删除当前指向的元素  
  40.     public boolean remove();  
  41.   
  42.     // 返回当前指向的元素  
  43.     public Object currentItem();  
  44. }  
  45.   
  46. public class ConcreteIterator implements Iterator {  
  47.     private Vector vector = new Vector();  
  48.     // 定义当前游标  
  49.     public int cursor = 0;  
  50.   
  51.     @SuppressWarnings("unchecked")  
  52.     public ConcreteIterator(Vector _vector) {  
  53.         this.vector = _vector;  
  54.     }  
  55.   
  56.     // 判断是否到达尾部  
  57.     public boolean hasNext() {  
  58.         if (this.cursor == this.vector.size()) {  
  59.             return false;  
  60.         } else {  
  61.             return true;  
  62.         }  
  63.     }  
  64.   
  65.     // 返回下一个元素  
  66.     public Object next() {  
  67.         Object result = null;  
  68.         if (this.hasNext()) {  
  69.             result = this.vector.get(this.cursor++);  
  70.         } else {  
  71.             result = null;  
  72.         }  
  73.         return result;  
  74.     }  
  75.   
  76.     // 删除当前元素  
  77.     public boolean remove() {  
  78.         this.vector.remove(this.cursor);  
  79.         return true;  
  80.     }  
  81.   
  82.     // 返回当前指向的元素  
  83.     public Object currentItem() {  
  84.         return this.vector.get(this.cursor);  
  85.     }  
  86. }  
  87.   
  88. public class Client {  
  89.     public static void main(String[] args) {  
  90.         // 声明出容器  
  91.         Aggregate agg = new ConcreteAggregate();  
  92.         // 产生对象数据放进去  
  93.         agg.add("abc");  
  94.         agg.add("aaa");  
  95.         agg.add("1234");  
  96.         // 遍历一下  
  97.         Iterator iterator = agg.iterator();  
  98.         while (iterator.hasNext()) {  
  99.             System.out.println(iterator.next());  
  100.         }  
  101.     }  
  102. }  


优点

 迭代器模式的主要优点如下:

       (1) 支持以不同的方式遍历一个聚合对象,在同一个聚合对象上可以定义多种遍历方式。在迭代器模式中只需要用一个不同的迭代器来替换原有迭代器即可改变遍历算法,我们也可以自己定义迭代器的子类以支持新的遍历方式。

       (2) 迭代器简化了聚合类。由于引入了迭代器,在原有的聚合对象中不需要再自行提供数据遍历等方法,这样可以简化聚合类的设计。

       (3) 在迭代器模式中,由于引入了抽象层,增加新的聚合类和迭代器类都很方便,无须修改原有代码,满足“开闭原则”的要求

缺点

    迭代器模式的主要缺点如下:

       (1) 由于迭代器模式将存储数据和遍历数据的职责分离,增加新的聚合类需要对应增加新的迭代器类,类的个数成对增加,这在一定程度上增加了系统的复杂性。

       (2) 抽象迭代器的设计难度较大,需要充分考虑到系统将来的扩展,例如JDK内置迭代器Iterator就无法实现逆向遍历,如果需要实现逆向遍历,只能通过其子类ListIterator等来实现,而ListIterator迭代器无法用于操作Set类型的聚合对象。在自定义迭代器时,创建一个考虑全面的抽象迭代器并不是件很容易的事情。

应用场景

  在以下情况下可以考虑使用迭代器模式:

       (1) 访问一个聚合对象的内容而无须暴露它的内部表示。将聚合对象的访问与内部数据的存储分离,使得访问聚合对象时无须了解其内部实现细节。

       (2) 需要为一个聚合对象提供多种遍历方式。

       (3) 为遍历不同的聚合结构提供一个统一的接口,在该接口的实现类中为不同的聚合结构提供不同的遍历方式,而客户端可以一致性地操作该接口。

注意事项

尽量不要自己写迭代器模式,看是否已提供。

扩展

6.1使用内部类实现迭代器

在迭代器模式结构图中,我们可以看到具体迭代器类和具体聚合类之间存在双重关系,其中一个关系为关联关系,在具体迭代器中需要维持一个对具体聚合对象的引用,该关联关系的目的是访问存储在聚合对象中的数据,以便迭代器能够对这些数据进行遍历操作。

       除了使用关联关系外,为了能够让迭代器可以访问到聚合对象中的数据,我们还可以将迭代器类设计为聚合类的内部类JDK中的迭代器类就是通过这种方法来实现的,如下AbstractList类代码片段所示:

[java]  view plain copy
  1. package java.util;  
  2. ……  
  3. public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {  
  4.     ......  
  5.     private class Itr implements Iterator<E> {  
  6.         int cursor = 0;  
  7.         ......  
  8. }  
  9. ……  
  10. }  

       我们可以通过类似的方法来设计第3节中的ProductList类,将ProductIterator类作为ProductList类的内部类,代码如下所示:

[java]  view plain copy
  1. //商品数据类:具体聚合类  
  2. class ProductList extends AbstractObjectList {  
  3.     public ProductList(List products) {  
  4.         super(products);  
  5.     }  
  6.       
  7.     public AbstractIterator createIterator() {  
  8.         return new ProductIterator();  
  9.     }  
  10.       
  11.     //商品迭代器:具体迭代器,内部类实现  
  12.     private class ProductIterator implements AbstractIterator {  
  13.         private int cursor1;  
  14.         private int cursor2;  
  15.           
  16.         public ProductIterator() {  
  17.             cursor1 = 0;  
  18.             cursor2 = objects.size() -1;  
  19.         }  
  20.           
  21.         public void next() {  
  22.             if(cursor1 < objects.size()) {  
  23.                 cursor1++;  
  24.             }  
  25.         }  
  26.           
  27.         public boolean isLast() {  
  28.             return (cursor1 == objects.size());  
  29.         }  
  30.           
  31.         public void previous() {  
  32.             if(cursor2 > -1) {  
  33.                 cursor2--;  
  34.             }  
  35.         }  
  36.           
  37.         public boolean isFirst() {  
  38.             return (cursor2 == -1);  
  39.         }  
  40.           
  41.         public Object getNextItem() {  
  42.             return objects.get(cursor1);  
  43.         }   
  44.               
  45.         public Object getPreviousItem() {  
  46.             return objects.get(cursor2);  
  47.         }     
  48.     }  
  49. }  

       无论使用哪种实现机制,客户端代码都是一样的,也就是说客户端无须关心具体迭代器对象的创建细节,只需通过调用工厂方法createIterator()即可得到一个可用的迭代器对象,这也是使用工厂方法模式的好处,通过工厂来封装对象的创建过程,简化了客户端的调用。


6.2JDK扩展迭代器

为了让开发人员能够更加方便地操作聚合对象,在JavaC#等编程语言中都提供了内置迭代器。在Java集合框架中,常用的ListSet等聚合类都继承(或实现)了java.util.Collection接口,在Collection接口中声明了如下方法(部分):

[java]  view plain copy
  1. package java.util;  
  2.   
  3. public interface Collection<E> extends Iterable<E> {  
  4.     ……  
  5. boolean add(Object c);  
  6. boolean addAll(Collection c);  
  7. boolean remove(Object o);  
  8. boolean removeAll(Collection c);  
  9. boolean remainAll(Collection c);   
  10. Iterator iterator();  
  11. ……  
  12. }  

       除了包含一些增加元素和删除元素的方法外,还提供了一个iterator()方法,用于返回一个Iterator迭代器对象,以便遍历聚合中的元素;具体的Java聚合类可以通过实现该iterator()方法返回一个具体的Iterator对象。

       JDK中定义了抽象迭代器接口Iterator,代码如下所示:

[java]  view plain copy
  1. package java.util;  
  2.   
  3. public interface Iterator<E> {  
  4. boolean hasNext();  
  5. E next();  
  6. void remove();  
  7. }  

       其中,hasNext()用于判断聚合对象中是否还存在下一个元素,为了不抛出异常,在每次调用next()之前需先调用hasNext(),如果有可供访问的元素,则返回truenext()方法用于将游标移至下一个元素,通过它可以逐个访问聚合中的元素,它返回游标所越过的那个元素的引用;remove()方法用于删除上次调用next()时所返回的元素。

       Java迭代器工作原理如图5所示,在第一个next()方法被调用时,迭代器游标由元素1”元素2”之间移至元素2”元素3”之间,跨越了元素2”,因此next()方法将返回对元素2”的引用;在第二个next()方法被调用时,迭代器由元素2”元素3”之间移至元素3”元素4”之间,next()方法将返回对元素3”的引用,如果此时调用remove()方法,即可将元素3”删除。

 

图5 Java迭代器示意图

       如下代码片段可用于删除聚合对象中的第一个元素:

[java]  view plain copy
  1. Iterator iterator = collection.iterator();   //collection是已实例化的聚合对象  
  2. iterator.next();        // 跳过第一个元素  
  3. iterator.remove();  // 删除第一个元素  

     需要注意的是,在这里,next()方法与remove()方法的调用是相互关联的。如果调用remove()之前,没有先对next()进行调用,那么将会抛出一个IllegalStateException异常,因为没有任何可供删除的元素。

     如下代码片段可用于删除两个相邻的元素:

[java]  view plain copy
  1. iterator.remove();  
  2. iterator.next();  //如果删除此行代码程序将抛异常  
  3. iterator.remove();    

       在上面的代码片段中如果将代码iterator.next();去掉则程序运行抛异常,因为第二次删除时将找不到可供删除的元素。

       在JDK中,Collection接口和Iterator接口充当了迭代器模式的抽象层,分别对应于抽象聚合类和抽象迭代器,而Collection接口的子类充当了具体聚合类,下面以List为例加以说明,图6列出了JDK中部分与List有关的类及它们之间的关系:

6 Java集合框架中部分类结构图

(注:为了简化类图,本图省略了大量方法)

       在JDK中,实际情况比图6要复杂很多,在图6中,List接口除了继承Collection接口的iterator()方法外,还增加了新的工厂方法listIterator(),专门用于创建ListIterator类型的迭代器,在List的子类LinkedList中实现了该方法,可用于创建具体的ListIterator子类ListItr的对象,代码如下所示:

[java]  view plain copy
  1. public ListIterator<E> listIterator(int index) {  
  2. return new ListItr(index);  
  3. }  

       listIterator()方法用于返回具体迭代器ListItr类型的对象。在JDK源码中,AbstractList中的iterator()方法调用了listIterator()方法,如下代码所示:

[java]  view plain copy
  1. public Iterator<E> iterator() {  
  2.     return listIterator();  
  3. }  

       客户端通过调用LinkedList类的iterator()方法,即可得到一个专门用于遍历LinkedList的迭代器对象。

      大家可能会问?既然有了iterator()方法,为什么还要提供一个listIterator()方法呢?这两个方法的功能不会存在重复吗?干嘛要多此一举?

      这是一个好问题,大笑。我给大家简单解释一下为什么要这样设计:由于在Iterator接口中定义的方法太少,只有三个,通过这三个方法只能实现正向遍历,而有时候我们需要对一个聚合对象进行逆向遍历等操作,因此在JDKListIterator接口中声明了用于逆向遍历的hasPrevious()previous()等方法,如果客户端需要调用这两个方法来实现逆向遍历,就不能再使用iterator()方法来创建迭代器了,因为此时创建的迭代器对象是不具有这两个方法的。我们只能通过如下代码来创建ListIterator类型的迭代器对象:

[java]  view plain copy
  1. ListIterator i = c.listIterator();  

       正因为如此,在JDKList接口中不得不增加对listIterator()方法的声明,该方法可以返回一个ListIterator类型的迭代器,ListIterator迭代器具有更加强大的功能。


[java]
 view plaincopy       在Java语言中,我们可以直接使用JDK内置的迭代器来遍历聚合对象中的元素,下面的代码演示了如何使用Java内置的迭代器:

  1. import java.util.*;  
  2.   
  3. class IteratorDemo {  
  4.    public static void process(Collection c) {  
  5.         Iterator i = c.iterator(); //创建迭代器对象  
  6.           
  7.         //通过迭代器遍历聚合对象  
  8.         while(i.hasNext()) {  
  9.             System.out.println(i.next().toString());  
  10.         }  
  11.    }  
  12.   
  13.     public static void main(String args[]) {  
  14.         Collection persons;  
  15. persons = new ArrayList(); //创建一个ArrayList类型的聚合对象  
  16.         persons.add("张无忌");  
  17.         persons.add("小龙女");  
  18.         persons.add("令狐冲");  
  19.         persons.add("韦小宝");  
  20.         persons.add("袁紫衣");  
  21.         persons.add("小龙女");  
  22.           
  23.         process(persons);  
  24.     }  
  25. }  

       在静态方法process()中使用迭代器IteratorCollection对象进行处理,该代码运行结果如下:

张无忌

小龙女

令狐冲

韦小宝

袁紫衣

小龙女

       如果需要更换聚合类型,如将List改成Set,则只需更换具体聚合类类名,如将上述代码中的ArrayList改为HashSet,则输出结果如下:

令狐冲

张无忌

韦小宝

小龙女

袁紫衣

       HashSet中合并了重复元素,并且元素以随机次序输出,其结果与使用ArrayList不相同。由此可见,通过使用迭代器模式,使得更换具体聚合类变得非常方便,而且还可以根据需要增加新的聚合类,新的聚合类只需要实现Collection接口,无须修改原有类库代码,符合“开闭原则”。


范例

7.1销售管理系统中数据的遍历

 Sunny软件公司为某商场开发了一套销售管理系统,在对该系统进行分析和设计时,Sunny软件公司开发人员发现经常需要对系统中的商品数据、客户数据等进行遍历,为了复用这些遍历代码,Sunny公司开发人员设计了一个抽象的数据集合类AbstractObjectList,而将存储商品和客户等数据的类作为其子类,AbstractObjectList类结构如图2所示:

2  AbstractObjectList类结构图

       在图2中,List类型的对象objects用于存储数据,方法说明如表1所示

1  AbstractObjectList类方法说明

方法名

方法说明

AbstractObjectList()

构造方法,用于给objects对象赋值

addObject()

增加元素

removeObject()

删除元素

getObjects()

获取所有元素

next()

移至下一个元素

isLast()

判断当前元素是否是最后一个元素

previous()

移至上一个元素

isFirst()

判断当前元素是否是第一个元素

getNextItem()

获取下一个元素

getPreviousItem()

获取上一个元素

       AbstractObjectList类的子类ProductListCustomerList分别用于存储商品数据和客户数据。

         Sunny软件公司开发人员通过对AbstractObjectList类结构进行分析,发现该设计方案存在如下几个问题:

        (1) 在图2所示类图中,addObject()removeObject()等方法用于管理数据,而next()isLast()previous()isFirst()等方法用于遍历数据。这将导致聚合类的职责过重,它既负责存储和管理数据,又负责遍历数据,违反了“单一职责原则”,由于聚合类非常庞大,实现代码过长,还将给测试和维护增加难度。

        (2) 如果将抽象聚合类声明为一个接口,则在这个接口中充斥着大量方法,不利于子类实现,违反了“接口隔离原则”。

        (3) 如果将所有的遍历操作都交给子类来实现,将导致子类代码庞大,而且必须暴露AbstractObjectList的内部存储细节,向子类公开自己的私有属性,否则子类无法实施对数据的遍历,这将破坏AbstractObjectList类的封装性。

       如何解决上述问题?解决方案之一就是将聚合类中负责遍历数据的方法提取出来,封装到专门的类中,实现数据存储和数据遍历分离,无须暴露聚合类的内部属性即可对其进行操作,而这正是迭代器模式的意图所在。

完整解决方案

       为了简化AbstractObjectList类的结构,并给不同的具体数据集合类提供不同的遍历方式,Sunny软件公司开发人员使用迭代器模式来重构AbstractObjectList类的设计,重构之后的销售管理系统数据遍历结构如图4所示:

图4 销售管理系统数据遍历结构图

(注:为了简化类图和代码,本结构图中只提供一个具体聚合类和具体迭代器类)

       在图4中,AbstractObjectList充当抽象聚合类,ProductList充当具体聚合类,AbstractIterator充当抽象迭代器,ProductIterator充当具体迭代器。完整代码如下所示:

[java]  view plain copy
  1. //在本实例中,为了详细说明自定义迭代器的实现过程,我们没有使用JDK中内置的迭代器,事实上,JDK内置迭代器已经实现了对一个List对象的正向遍历  
  2. import java.util.*;  
  3.   
  4. //抽象聚合类  
  5. abstract class AbstractObjectList {  
  6.     protected List<Object> objects = new ArrayList<Object>();  
  7.   
  8.     public AbstractObjectList(List objects) {  
  9.         this.objects = objects;  
  10.     }  
  11.       
  12.     public void addObject(Object obj) {  
  13.         this.objects.add(obj);  
  14.     }  
  15.       
  16.     public void removeObject(Object obj) {  
  17.         this.objects.remove(obj);  
  18.     }  
  19.       
  20.     public List getObjects() {  
  21.         return this.objects;  
  22.     }  
  23.       
  24.     //声明创建迭代器对象的抽象工厂方法  
  25.     public abstract AbstractIterator createIterator();  
  26. }  
  27.   
  28. //商品数据类:具体聚合类  
  29. class ProductList extends AbstractObjectList {  
  30.     public ProductList(List products) {  
  31.         super(products);  
  32.     }  
  33.       
  34.     //实现创建迭代器对象的具体工厂方法  
  35.     public AbstractIterator createIterator() {  
  36.         return new ProductIterator(this);  
  37.     }  
  38. }   
  39.   
  40. //抽象迭代器  
  41. interface AbstractIterator {  
  42.     public void next(); //移至下一个元素  
  43.     public boolean isLast(); //判断是否为最后一个元素  
  44.     public void previous(); //移至上一个元素  
  45.     public boolean isFirst(); //判断是否为第一个元素  
  46.     public Object getNextItem(); //获取下一个元素  
  47.     public Object getPreviousItem(); //获取上一个元素  
  48. }  
  49.   
  50. //商品迭代器:具体迭代器  
  51. class ProductIterator implements AbstractIterator {  
  52.     private ProductList productList;  
  53.     private List products;  
  54.     private int cursor1; //定义一个游标,用于记录正向遍历的位置  
  55.     private int cursor2; //定义一个游标,用于记录逆向遍历的位置  
  56.       
  57.     public ProductIterator(ProductList list) {  
  58.         this.productList = list;  
  59.         this.products = list.getObjects(); //获取集合对象  
  60.         cursor1 = 0//设置正向遍历游标的初始值  
  61.         cursor2 = products.size() -1//设置逆向遍历游标的初始值  
  62.     }  
  63.       
  64.     public void next() {  
  65.         if(cursor1 < products.size()) {  
  66.             cursor1++;  
  67.         }  
  68.     }  
  69.       
  70.     public boolean isLast() {  
  71.         return (cursor1 == products.size());  
  72.     }  
  73.       
  74.     public void previous() {  
  75.         if (cursor2 > -1) {  
  76.             cursor2--;  
  77.         }  
  78.     }  
  79.       
  80.     public boolean isFirst() {  
  81.         return (cursor2 == -1);  
  82.     }  
  83.       
  84.     public Object getNextItem() {  
  85.         return products.get(cursor1);  
  86.     }   
  87.           
  88.     public Object getPreviousItem() {  
  89.         return products.get(cursor2);  
  90.     }     
  91. }  

       编写如下客户端测试代码:

[java]  view plain copy
  1. class Client {  
  2.     public static void main(String args[]) {  
  3.         List products = new ArrayList();  
  4.         products.add("倚天剑");  
  5.         products.add("屠龙刀");  
  6.         products.add("断肠草");  
  7.         products.add("葵花宝典");  
  8.         products.add("四十二章经");  
  9.               
  10.         AbstractObjectList list;  
  11.         AbstractIterator iterator;  
  12.           
  13.         list = new ProductList(products); //创建聚合对象  
  14.         iterator = list.createIterator();   //创建迭代器对象  
  15.           
  16.         System.out.println("正向遍历:");      
  17.         while(!iterator.isLast()) {  
  18.             System.out.print(iterator.getNextItem() + ",");  
  19.             iterator.next();  
  20.         }  
  21.         System.out.println();  
  22.         System.out.println("-----------------------------");  
  23.         System.out.println("逆向遍历:");  
  24.         while(!iterator.isFirst()) {  
  25.             System.out.print(iterator.getPreviousItem() + ",");  
  26.             iterator.previous();  
  27.         }  
  28.     }  
  29. }  

       编译并运行程序,输出结果如下:

正向遍历:

倚天剑,屠龙刀,断肠草,葵花宝典,四十二章经,

-----------------------------

逆向遍历:

四十二章经,葵花宝典,断肠草,屠龙刀,倚天剑

       如果需要增加一个新的具体聚合类,如客户数据集合类,并且需要为客户数据集合类提供不同于商品数据集合类的正向遍历和逆向遍历操作,只需增加一个新的聚合子类和一个新的具体迭代器类即可,原有类库代码无须修改,符合“开闭原则”;如果需要为ProductList类更换一个迭代器,只需要增加一个新的具体迭代器类作为抽象迭代器类的子类,重新实现遍历方法,原有迭代器代码无须修改,也符合“开闭原则”;但是如果要在迭代器中增加新的方法,则需要修改抽象迭代器源代码,这将违背“开闭原则”。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值