004-行为型-04-迭代器模式(Iterator)

1、概述

  提供一种方法顺序访问一个聚合对象中各个元素, 而又无须暴露该对象的内部表示。java

  迭代器模式就是分离了集合对象的遍历行为,抽象出一个迭代器类来负责,这样既能够作到不暴露集合的内部结构,又可以让外部代码透明地访问集合内部的数据。apache

  迭代模式使用比较少,JDK集合也提供了Iterator的具体实现,能够直接拿来用,没必要本身实现编程

1.一、适用场景

  一、访问一个聚合对象的内容而无须暴露它的内部表示。设计模式

  二、须要为聚合对象提供多种遍历方式。数组

  三、为遍历不一样的聚合结构提供一个统一的接口。mybatis

1.二、优缺点

优势: 一、它支持以不一样的方式遍历一个聚合对象。 二、迭代器简化了聚合类。 三、在同一个聚合上能够有多个遍历。 四、在迭代器模式中,增长新的聚合类和迭代器类都很方便,无须修改原有代码。ide

缺点:因为迭代器模式将存储数据和遍历数据的职责分离,增长新的聚合类须要对应增长新的迭代器类,类的个数成对增长,这在必定程度上增长了系统的复杂性。测试

1.三、类图角色及其职责this

   

  迭代模式的角色和职责

  一、Iterator(迭代器接口)spa

  该接口必须定义实现迭代功能的最小定义方法集

  好比提供hasNext()和next()方法。

  二、ConcreteIterator(迭代器实现类)

  迭代器接口Iterator的实现类。能够根据具体状况加以实现。

  三、Aggregate(容器接口)

  定义基本功能以及提供相似Iterator iterator()的方法。

  四、concreteAggregate(容器实现类)

  容器接口的实现类。必须实现Iterator iterator()方法。

1.四、演进过程

1.4.一、初始化

建立一个容器中要存储的内容Book

public class Book {
    private String id;
    private String name;
    private double price;

    public Book(String id, String name, double price) {
        this.id = id;
        this.name = name;
        this.price = price;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    public void display() {
        System.out.println("ID=" + id + ",\tname=" + name + ",\tprice" + price);
    }
}
View Code

 

建立一个容器BookList

public class BookList {
    //容器内部仍是一个List,也能够用数组
    private List<Book> bookList = new ArrayList<Book>();
    private int index;

    //添加书籍
    public void addBook(Book book){
        bookList.add(book);
    }

    //删除书籍
    public void removeBook(Book book){
        int bookIndex = bookList.indexOf(book);
        bookList.remove(bookIndex);
    }

    //判断是否有下一本书
    public boolean hasNext(){
        if(index >= bookList.size()){
            return false;
        }
        return true;
    }

    //得到下一本书
    public Book getNext(){
        return bookList.get(index++);
    }

    //获取集合长度
    public int getSize(){
        return bookList.size();
    }

    //根据index获取Book
    public Book getByIndex(int index){
        return bookList.get(index);
    }
}

  接下来,就是迭代容器了

方式1、(由容器本身实现顺序遍历。直接在容器类里直接添加顺序遍历方法)

    @Test
    public void test1() {
        BookList bookList = new BookList();

        Book book1 = new Book("001","设计模式",200);
        Book book2 = new Book("002","Java核心编程",200);
        Book book3 = new Book("003","计算机组成原理",200);

        bookList.addBook(book1);
        bookList.addBook(book2);
        bookList.addBook(book3);

        while(bookList.hasNext()){
            Book book = bookList.getNext();
            book.display();
        }
    }

输出

ID=001,    name=设计模式,    price200.0
ID=002,    name=Java核心编程,    price200.0
ID=003,    name=计算机组成原理,    price200.0

 

方式2、让调用者本身实现遍历。直接暴露数据细节给外部)

    @Test
    public void test2() {
        BookList bookList = new BookList();

        Book book1 = new Book("001", "设计模式", 200);
        Book book2 = new Book("002", "Java核心编程", 200);
        Book book3 = new Book("003", "计算机组成原理", 200);

        bookList.addBook(book1);
        bookList.addBook(book2);
        bookList.addBook(book3);

        for (int i = 0; i < bookList.getSize(); i++) {
            Book book = bookList.getByIndex(i);
            book.display();
        }
    }

结果同上

不使用迭代模式的缺点

  以上方法1与方法2均可以实现对遍历,但有这些问题

  1,容器类承担了太多功能:一方面须要提供添加删除等自己应有的功能;一方面还须要提供遍历访问功能。

  2,每每容器在实现遍历的过程当中,须要保存遍历状态,当跟元素的添加删除等功能夹杂在一块儿,很容易引发混乱和程序运行错误等。

应用迭代模式的条件

  Iterator模式就是为了有效地处理按顺序进行遍历访问的一种设计模式,简单地说,Iterator模式提供一种有效的方法,能够屏蔽汇集对象集合的容器类的实现细节,而能对容器内包含的对象元素按顺序进行有效的遍历访问。

  因此,Iterator模式的应用场景能够概括为知足如下几个条件:

  一、访问容器中包含的内部对象
  二、按顺序访问

1.4.二、进化

用代码实现一下迭代模式,只需修改BookList便可,BookListIterator

public class BookListIterator {
    //容器内部仍是一个List,也能够用数组
    private List<Book> bookList = new ArrayList<Book>();
    private int index;

    //添加书籍
    public void addBook(Book book){
        bookList.add(book);
    }

    //删除书籍
    public void removeBook(Book book){
        int bookIndex = bookList.indexOf(book);
        bookList.remove(bookIndex);
    }

    //获取集合长度
    public int getSize(){
        return bookList.size();
    }

    //根据index获取Book
    public Book getByIndex(int index){
        return bookList.get(index);
    }

    //获得Iterator实例
    public Iterator Iterator() {
        return new Itr();
    }

    //内部类,Iterator实例(由于要使用容器的内部信息,因此要写成内部类)
    private class Itr implements Iterator{
        //判断是否有下一本书,将刚才hasNext()中内容复制过来便可
        public boolean hasNext() {
            if(index >= bookList.size()){
                return false;
            }
            return true;
        }
        //得到下一本书,将刚才getNext()中内容复制过来便可
        public Object next() {
            return bookList.get(index++);
        }

        public void remove() {

        }
    }
}

测试

    @Test
    public void test3() {
        BookListIterator bookList = new BookListIterator();

        Book book1 = new Book("001","设计模式",200);
        Book book2 = new Book("002","Java核心编程",200);
        Book book3 = new Book("003","计算机组成原理",200);

        bookList.addBook(book1);
        bookList.addBook(book2);
        bookList.addBook(book3);

        Iterator iterator = bookList.Iterator();
        while(iterator.hasNext()){
            Book book = (Book) iterator.next();
            book.display();
        }
    }

输出同上

  能够看到,这和使用JDK提供集合的Iterator方法就如出一辙了。

2、扩展

2.1 JDK

java.util.Iterator
java.util.ArrayList中的Itr

2.2 Mybatis

org.apache.ibatis.cursor.defaults.DefaultCursor的cursorIterator

 

 

 

c

相关文章
相关标签/搜索