迭代器模式

简介: 1、什么是迭代器呢?   迭代器是一种取出元素的方式,也就是遍历某一元素集合内元素的方式。我们平常对于数组的遍历方式是通过下角标利用for循环等来操作数据,元素集合不止数组一种,例如还有ArrayList(底层为数组)、LinkedList(底层为链表)、HashSet(使用hash表)等,每一个集合内部的存储结构都是不同的,所以每一个集合存和取都是不一样,那么针对这些集合遍历就要分别使

1、什么是迭代器呢?

  迭代器是一种取出元素的方式,也就是遍历某一元素集合内元素的方式。我们平常对于数组的遍历方式是通过下角标利用for循环等来操作数据,元素集合不止数组一种,例如还有ArrayList(底层为数组)、LinkedList(底层为链表)、HashSet(使用hash表)等,每一个集合内部的存储结构都是不同的,所以每一个集合存和取都是不一样,那么针对这些集合遍历就要分别使用对应的遍历方式。


  那么,有没有一个统一的解决方案来处理这样的循环遍历问题呢?我们只提供一种模式就能够遍历所有的不同数据结构的集合,而且还不能够暴露集合内部的表示(即不用知道内部实现细节到底是数组还是ArrayList、LinkedList,这样就不必在外部应用中暴露获得元素的方式),这就需要迭代器设计模式闪亮登场了。


  迭代器模式的出现统一了数据循环遍历的问题,让我们不再关心软件系统内部的实现细节,按照统一的模式顺序访问一个集合对象中的各个元素。


2、迭代器模式的方式?


  因为每个集合对象本身不一样,所以当我们对不同类型的集合进行遍历时,由于必须暴露其内部表示,因此所写的遍历语句(for)是不能通用的。如果我们对需要用到的集合创建一个迭代器,由于对迭代器的遍历语句是相同的,那么就可以实现遍历语句的通用性。另外,拥有了迭代器就意味着我们对集合的实现方式是完全不知道的,即不必知道集合内部的实现细节,以后一旦需要改变集合的实现,例如从数组改为ArrayList,客户端的代码是不需要发生任何改变的,甚至于客户端根本就不知道所遍历的集合发生了改变。


  对于遍历的方式,首先谈一下Java语言内置的迭代器,即集合中的迭代器,其是耦合在集合对象内部,让容器实现遍历功能,这种方式应用在Java集合的底层代码实现中;


  Java语言内置的迭代器是在外面定义一个迭代器的接口Iterator,这样所有的集合在其内部定义迭代器时统一实现该接口,这样所有集合内部都有了自己的迭代器,单列集合的根接口Collection中定义一个获取迭代器的方法public Iterator iterator()专门用于获取其迭代器对象,这样所有实现Collection接口的实现类中只需要重写该iterator方法,返回自己的迭代器对象即可。


  这种方式存在一定的弊端:首先,将遍历算法耦合在集合对象内部,这就加重了几何对象的负担,不但需要处理数据的添加、删除和修改,还要负责数据的遍历,显然,集合对象的负担过重,这也是不符合单一职责原则的。


  另外一种方式就是今天我们需要讲的一种方式:




  这需要满足以下两点内容:

    (1)迭代器在不需要暴露集合对象内部实现细节的情况下,能够遍历集合中的数据;

    (2)迭代器获取集合对象内部数据的方式,也不需要依赖于集合对象下标,即生成迭代器之后,就完全脱离集合对象的约束。


  在迭代器的遍历过程中,我们没有暴露集合类的内部细节内容,而是返回object类型,而遍历方法也是不需要任何下标信息的,因为我们可以在迭代器内部获得集合对象的引用,可以默认下标从-1开始,在遍历方法中顺序变化,从而获得集合对象内部元素内容。这样,就有效的将迭代器与集合对象分离开,使两者从产生迭代器之后就各不相干,而客户端调用只与迭代器遍历相关,不再依赖于集合对象的内部细节。


  迭代器建模图:




3、迭代器模式案例?


  需求:对人员信息使用迭代器模式进行遍历。

  (1)首先得到人员信息接口,里面只有一个获得人员信息的方法:


<span style="font-size:18px;">package 迭代器模式;
/**
 * 需求:人员信息接口
 * 		方法:获得人员信息
 * @author win2016
 *
 */
public interface IPerson {
	//获得人员信息方法
	public String getPersonInfo();
}</span>

  (2)人员信息类,用于存储人员的基本信息:


<span style="font-size:18px;">package 迭代器模式;
/**
 * 需求:人员类
 * 
 * @author win2016
 *
 */
public class Person implements IPerson {
	//姓名
	private String name;
	//年龄
	private int age;
	//性别
	private int sex;
	//构造方法传入值
	public Person(String name,int age,int sex){
		this.name = name;
		this.age = age;
		this.sex = sex;
	}
	//获取人员信息
	@Override
	public String getPersonInfo() {
		return "姓名:"+this.name+"-年龄"+this.age+"-性别"+(this.sex==1?"男":(this.sex==0?"女":""));
	}
	
}</span>


  (3)人员集合接口,有两个方法,一个是获得人员内部信息列表方法,一个是获得迭代器接口方法,设置两种方法是为了在客户端应用中分别使用,便于比较。


<span style="font-size:18px;">package 迭代器模式;

import java.util.ArrayList;

/**
 * 需求:人员集合接口 获得内部存储人员信息内容
 * 		方法:获得人员内部信息列表方法;获得迭代器接口方法(目的:在客户端应用中分别使用)
 * @author win2016
 *
 */
public interface IPersonList {
	//获得内部存储人员信息内容
	public ArrayList<IPerson> getPersonList();
	//迭代器
	public Iterator1 iterator();
}</span>

  (4)人员集合实现:


<span style="font-size:18px;">package 迭代器模式;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.Random;

/**
 * 需求:人员集合实现类
 * 		在PersonList的构造方法中初始化人员信息列表内容
 * 
 * @author win2016
 *
 */
public class PersonList implements IPersonList {
	//存储用户信息集合
	private ArrayList<IPerson> list = new ArrayList<IPerson>();
	//构造方法初始化人员信息
	public PersonList(){
		Random random = new Random();
		//创建人员信息
		for (int i = 0; i < 6; i++) {
			IPerson person = new Person("人员"+i,random.nextInt(30),random.nextInt(2));
			//存入集合
			list.add(person);
		}
	}
	//获得内部人员信息
	public ArrayList<IPerson> getPersonList(){
		return this.list;
	} 
	//迭代器
	public Iterator1 iterator(){
		return new PersonIterator(this.list);
	}
}</span>

  (5)迭代器接口:


<span style="font-size:18px;">package 迭代器模式;
/**
 * 需求:迭代器接口
 * 方法:判断是否存在下一个元素;获取下一个元素方法;删除方法
 * @author win2016
 *
 */
public interface Iterator1 {
	//判断是否有下一个节点
	public boolean hasNext();
	//获得下一个节点对象
	public Object next();
	//删除方法
	public Object remove();
}</span>

  (6)迭代器实现类:


<span style="font-size:18px;">package 迭代器模式;

import java.util.ArrayList;

/**
 * 需求:迭代器实现类
 * 
 * @author win2016
 *
 */
public class PersonIterator implements Iterator1 {
	//存储人员列表对象信息
	private final ArrayList<IPerson> personList;
	//存储位置信息,初始值为-1
	private int index = -1;
	//构造方法将人员列表对象传入
	public PersonIterator(ArrayList<IPerson> personList){
		this.personList = personList;
	}
	//是否有下一个元素
	@Override
	public boolean hasNext() {
		return (this.personList == null? false:(index<this.personList.size()-1));
	}
	//获得下一个元素对象
	@Override
	public Object next() {
		return this.personList.get(++index);
	}
	//删除对象
	@Override
	public Object remove() {
		if (this.personList != null) {
			return this.personList.remove(index);
		}
		return null;
	}

}</span>


  (7)测试类:


<span style="font-size:18px;">package 迭代器模式;

/**
 * 需求:测试类
 * @author win2016
 *
 */
public class Test {
	public static void main(String[] args) {
		//创建人员列表对象
		IPersonList personList = new PersonList1();
		System.out.println("--------使用迭代器输出人员信息----------");
		//生成迭代器
		Iterator1 iterator = personList.iterator();
		//循环迭代器,遍历每一个元素输出人员信息
		while(iterator.hasNext()){
			//获得人员对象实例
			IPerson person = (IPerson) iterator.next();
			if (person!=null) {
				//输出人员信息
				System.out.println(person.getPersonInfo());
			}
		}
	}
}</span>

  上面是使用迭代器迭代了ArrayList集合。

  那么,迭代器如何面对变化呢?

    现在我们假设人员列表信息类型改变了,使用数组类型而不是ArrayList,该如何处理呢?我们只要增加一个用数组实现的PersonList1:


<span style="font-size:18px;">package 迭代器模式;

import java.util.ArrayList;
import java.util.Random;

/**
 * 需求:数组类型
 * @author win2016
 *
 */
public class PersonList1 implements IPersonList {
	//存储用户信息列表
	private final IPerson[] list = new IPerson[6];
	//构造方法初始化人员信息
	public PersonList1(){
		Random random = new Random();
		//创建人员信息
		for (int i = 0; i < 6; i++) {
			IPerson person = new Person("人员"+i, random.nextInt(30), random.nextInt(2));
			list[i] = person;
		}
	}
	//迭代器
	public Iterator1 iterator(){
		return new ArrPersonIterator(this.list);
	}
	//获得内部存储人员信息内容
	public IPerson[] getPersonInfo() {
		return list;
	}
	@Override
	public ArrayList<IPerson> getPersonList() {
		// TODO Auto-generated method stub
		return null;
	}

}</span>

    然后再增加一个数组迭代器:


<span style="font-size:18px;">package 迭代器模式;
/***
 * 需求:数组迭代器
 * 
 * @author win2016
 *
 */
public class ArrPersonIterator implements Iterator1 {
	//私有属性存储人员列表对象信息
	private final IPerson[] personList;
	//存储位置信息,初始值为-1
	private int index = -1;
	//构造方法将人员列表对象存入
	public ArrPersonIterator(IPerson[] personList){
		this.personList = personList;
	}
	//判断是否有下一个节点
	public boolean hasNext(){
		return (this.personList == null?false:(index<this.personList.length-1));
	}
	//获取下一个节点元素
	public Object next(){
		if (this.personList!=null&&(index<this.personList.length-1)) {
			//获得人员列表对象中的人员信息
			return this.personList[++index];
		}
		return null;
	}
	//删除对象
	public Object remove(){
		if (this.personList!=null) {
			IPerson person = this.personList[index];
			this.personList[index] = null;
			return person;
		}
		return null;
	}
}</span>

  这样就可以啦。


  总结:当你需要访问一个聚合对象,而且不管这些对象是什么都需要遍历的时候,就应该考虑使用迭代器模式。另外,当需要对聚集有多种方式遍历时,可以考虑去使用迭代器模式。迭代器模式为遍历不同的聚集结构提供如开始、下一个、是否结束、当前哪一项等统一的接口。




目录
相关文章
|
1月前
|
设计模式 编译器 C#
访问者模式
访问者模式是一种行为型设计模式,用于将数据结构与其操作解耦。通过在不改变数据结构的前提下增加新的操作,访问者模式提供了一种灵活的方式来实现功能扩展。其关键特性包括分离操作、支持扩展和双分派机制。适用于需要对对象结构中的元素执行多种操作、频繁扩展操作以及不希望操作逻辑和数据结构耦合的场景。常见的使用案例包括编译器中的语法树、文件系统和游戏场景。优点是增加操作时无需修改数据结构,符合开闭原则;缺点是添加新的数据结构类型时需要修改所有访问者类,维护成本较高。
48 3
|
8月前
|
设计模式 uml C++
行为型 迭代器模式
行为型 迭代器模式
49 0
|
设计模式 存储 算法
设计模式~迭代器模式(Iterator)-20
迭代器模式(Iterator Pattern)是Java和.Net编程环境中非常常用的设计模式。这种模式用于顺序访问集合对象的元素,不需要知道集合对象的底层表示。迭代器模式属于行为型模式。迭代器模式已经被淘汰,java中已经把迭代器运用到各个聚集类(collection)中了,使用java自带的迭代器就已经满足我们的需求了 目录 迭代器模式(Iterator) (1)优点 (2)缺点 (3)使用场景 (4)注意事项 (5)应用实例: 代码
63 0
|
设计模式 存储 Java
迭代器模式
迭代器模式(Iterator Pattern)是一种行为型设计模式,它提供了一种顺序访问聚合对象中每个元素的方法,而不暴露其内部实现。
108 1
|
容器
关于迭代器模式我所知道的
关于迭代器模式我所知道的
61 0
今天说说你不知道的访问者模式
今天说说你不知道的访问者模式
84 0
|
JavaScript 前端开发
简单理解迭代器模式
这几天研究了一下迭代器模式,期间有一段时间经常搞不太懂一些概念与概念之间的关系,今天来整理一下。
134 0
|
设计模式 存储 容器
我学会了,迭代器模式
迭代器模式属于行为型模式,这个类型的设计模式总结出了 类、对象之间的经典交互方式,将类、对象的行为和使用解耦了,花式的去使用对象的行为来完成特定场景下的功能。
109 0
我学会了,迭代器模式
|
Java 容器
迭代器模式
迭代器模式
117 0
|
Java uml 容器
被用到炉火纯清的迭代器模式
迭代器模式(Iterator),提供一种方法顺序访问一个聚合对象中的各种元素,而又不暴露该对象的内部表示。
120 0
被用到炉火纯清的迭代器模式