本文转载地址:

           http://blog.csdn.net/chenssy/article/details/37521461


迭代对于我们搞Java的来说绝对不陌生。我们常常使用JDK提供的迭代接口进行Java集合的迭代。

[java] view plaincopyprint?
  1. Iterator iterator = list.iterator();
  2. while(iterator.hasNext()){
  3. String string = iterator.next();
  4. //do something
  5. }

迭代其实我们可以简单地理解为遍历,是一个标准化遍历各类容器里面的所有对象的方法类,它是一个很典型的设计模式。Iterator模式是用于遍历集合类的标准访问方法。它可以把访问逻辑从不同类型的集合类中抽象出来,从而避免向客户端暴露集合的内部结构。 在没有迭代器时我们都是这么进行处理的。如下:

对于数组我们是使用下标来进行处理的:

[java] view plaincopyprint?
  1. int[] arrays = new int[10];
  2. for(int i = 0 ; i < arrays.length ; i++){
  3. int a = arrays[i];
  4. //do something
  5. }

对于ArrayList是这么处理的:

[java] view plaincopyprint?
  1. List<String> list = new ArrayList<String>();
  2. for(int i = 0 ; i < list.size() ;  i++){
  3. String string = list.get(i);
  4. //do something
  5. }

对于这两种方式,我们总是都事先知道集合的内部结构,访问代码和集合本身是紧密耦合的,无法将访问逻辑从集合类和客户端代码中分离出来。同时每一种集合对应一种遍历方法,客户端代码无法复用。 在实际应用中如何需要将上面将两个集合进行整合是相当麻烦的。所以为了解决以上问题,Iterator模式腾空出世,它总是用同一种逻辑来遍历集合。使得客户端自身不需要来维护集合的内部结构,所有的内部状态都由Iterator来维护。客户端从不直接和集合类打交道,它总是控制Iterator,向它发送"向前","向后","取当前元素"的命令,就可以间接遍历整个集合。

上面只是对Iterator模式进行简单的说明,下面我们看看Java中Iterator接口,看他是如何来进行实现的。

一、java.util.Iterator

在Java中Iterator为一个接口,它只提供了迭代了基本规则,在JDK中他是这样定义的:对 collection 进行迭代的迭代器。迭代器取代了 Java Collections Framework 中的 Enumeration。迭代器与枚举有两点不同:

1、迭代器允许调用者利用定义良好的语义在迭代期间从迭代器所指向的 collection 移除元素。

2、方法名称得到了改进。

其接口定义如下:

[java] view plaincopyprint?
  1. public interface Iterator {
  2.   boolean hasNext();
  3.   Object next();
  4.   void remove();
  5. }

其中:

Object next():返回迭代器刚越过的元素的引用,返回值是Object,需要强制转换成自己需要的类型

boolean hasNext():判断容器内是否还有可供访问的元素

void remove():删除迭代器刚越过的元素

对于我们而言,我们只一般只需使用next()、hasNext()两个方法即可完成迭代。如下:

[java] view plaincopyprint?
  1. for(Iterator it = c.iterator(); it.hasNext(); ) {
  2.   Object o = it.next();
  3.    //do something
  4. }

前面阐述了Iterator有一个很大的优点,就是我们不必知道集合的内部结果,集合的内部结构、状态由Iterator来维持,通过统一的方法hasNext()、next()来判断、获取下一个元素,至于具体的内部实现我们就不用关心了。但是作为一个合格的程序员我们非常有必要来弄清楚Iterator的实现。下面就ArrayList的源码进行分析分析。

二、各个集合的Iterator的实现

下面就ArrayList的Iterator实现来分析,其实如果我们理解了ArrayList、Hashset、TreeSet的数据结构,内部实现,对于他们是如何实现Iterator也会胸有成竹的。因为ArrayList的内部实现采用数组,所以我们只需要记录相应位置的索引即可,其方法的实现比较简单。

2.1、ArrayList的Iterator实现

在ArrayList内部首先是定义一个内部类Itr,该内部类实现Iterator接口,如下:

[java] view plaincopyprint?
  1. private class Itr implements Iterator<E> {
  2. //do something
  3. }

而ArrayList的iterator()方法实现:

[java] view plaincopyprint?
  1. public Iterator<E> iterator() {
  2. return new Itr();
  3. }

所以通过使用ArrayList.iterator()方法返回的是Itr()内部类,所以现在我们需要关心的就是Itr()内部类的实现:

在Itr内部定义了三个int型的变量:cursor、lastRet、expectedModCount。其中cursor表示下一个元素的索引位置,lastRet表示上一个元素的索引位置

[java] view plaincopyprint?
  1. int cursor;
  2. int lastRet = -1;
  3. int expectedModCount = modCount;

从cursor、lastRet定义可以看出,lastRet一直比cursor少一所以hasNext()实现方法异常简单,只需要判断cursor和lastRet是否相等即可。

[java] view plaincopyprint?
  1. public boolean hasNext() {
  2. return cursor != size;
  3. }

对于next()实现其实也是比较简单的,只要返回cursor索引位置处的元素即可,然后修改cursor、lastRet即可,

[java] view plaincopyprint?
  1. public E next() {
  2. checkForComodification();
  3. int i = cursor;    //记录索引位置
  4. if (i >= size)    //如果获取元素大于集合元素个数,则抛出异常
  5. throw new NoSuchElementException();
  6. Object[] elementData = ArrayList.this.elementData;
  7. if (i >= elementData.length)
  8. throw new ConcurrentModificationException();
  9. cursor = i + 1;      //cursor + 1
  10. return (E) elementData[lastRet = i];  //lastRet + 1 且返回cursor处元素
  11. }

checkForComodification()主要用来判断集合的修改次数是否合法,即用来判断遍历过程中集合是否被修改过。在java提高篇(二一)-----ArrayList中已经阐述了。modCount用于记录ArrayList集合的修改次数,初始化为0,,每当集合被修改一次(结构上面的修改,内部update不算),如add、remove等方法,modCount + 1,所以如果modCount不变,则表示集合内容没有被修改。该机制主要是用于实现ArrayList集合的快速失败机制,在Java的集合中,较大一部分集合是存在快速失败机制的,这里就不多说,后面会讲到。所以要保证在遍历过程中不出错误,我们就应该保证在遍历过程中不会对集合产生结构上的修改(当然remove方法除外),出现了异常错误,我们就应该认真检查程序是否出错而不是catch后不做处理。

[java] view plaincopyprint?
  1. final void checkForComodification() {
  2. if (modCount != expectedModCount)
  3. throw new ConcurrentModificationException();
  4. }

对于remove()方法的是实现,它是调用ArrayList本身的remove()方法删除lastRet位置元素,然后修改modCount即可。

[java] view plaincopyprint?
  1. public void remove() {
  2. if (lastRet < 0)
  3. throw new IllegalStateException();
  4. checkForComodification();
  5. try {
  6. ArrayList.this.remove(lastRet);
  7. cursor = lastRet;
  8. lastRet = -1;
  9. expectedModCount = modCount;
  10. } catch (IndexOutOfBoundsException ex) {
  11. throw new ConcurrentModificationException();
  12. }
  13. }

这里就对ArrayList的Iterator实现讲解到这里,对于Hashset、TreeSet等集合的Iterator实现,各位如果感兴趣可以继续研究,个人认为在研究这些集合的源码之前,有必要对该集合的数据结构有清晰的认识,这样会达到事半功倍的效果!!!!

转载于:https://www.cnblogs.com/hthuang/p/4418677.html

java提高篇(三十)-----Iterator相关推荐

  1. java迭代器cas,java提高篇(三十)-Iterator - Java 技术驿站-Java 技术驿站

    迭代对于我们搞Java的来说绝对不陌生.我们常常使用JDK提供的迭代接口进行Java集合的迭代. Iterator iterator = list.iterator(); while(iterator ...

  2. 【转】java提高篇(十)-----详解匿名内部类

    原文网址:http://www.cnblogs.com/chenssy/p/3390871.html 在java提高篇-----详解内部类中对匿名内部类做了一个简单的介绍,但是内部类还存在很多其他细节 ...

  3. java提高篇(三)-----理解java的三大特性之多态

    面向对象编程有三大特性:封装.继承.多态. 封装隐藏了类的内部实现机制,可以在不影响使用的情况下改变类的内部结构,同时也保护了数据.对外界而已它的内部细节是隐藏的,暴露给外界的只是它的访问方法. 继承 ...

  4. Java Review(三十六、IO)

    文章目录 File 类 访问文件和目录 1.访问文件名相关的方法 2. 文件检测相关的方法 3. 获取常规文件信息 4. 文件操作相关的方法 5. 目录操作相关的方法 文件过滤器 IO流概览 流的分类 ...

  5. Java提高篇(三四)-----fail-fast机制

    Java提高篇(三四)-----fail-fast机制 在JDK的Collection中我们时常会看到类似于这样的话: 例如,ArrayList: 注意,迭代器的快速失败行为无法得到保证,因为一般来说 ...

  6. java提高篇(八)----详解内部类

    可以将一个类的定义放在另一个类的定义内部,这就是内部类. 内部类是一个非常有用的特性但又比较难理解使用的特性(鄙人到现在都没有怎么使用过内部类,对内部类也只是略知一二). 第一次见面 内部类我们从外面 ...

  7. java提高篇之数组(2)

    前面一节主要介绍了数组的基本概念,对什么是数组稍微深入了一点点,在这篇博文中主要介绍数组的其他方面. 三.性能?请优先考虑数组 在java中有很多方式来存储一系列数据,而且在操作上面比数组方便的多?但 ...

  8. java comparable接口_Java面试题之Java集合篇三

    Java面试题之Java集合篇三1.HashMap和HashTable有何不同? (1)HashMap允许key和value为null,而HashTable不允许. (2)HashTable是同步的, ...

  9. java提高篇之详解内部类

    转载自 java提高篇之详解内部类 内部类是一个非常有用的特性但又比较难理解使用的特性(鄙人到现在都没有怎么使用过内部类,对内部类也只是略知一二). 第一次见面 内部类我们从外面看是非常容易理解的,无 ...

最新文章

  1. pytorch 常用问题解决
  2. poj2392 Space Elevator
  3. mui初级入门教程(六)— 模板页面实现原理及多端适配指南
  4. include引入php报错,如何解决引入php文件报错的问题
  5. 如何设置共享Mac上的视频、音乐或照片的教程
  6. 线程(一)__同步以及死锁
  7. NameNode故障处理之数据恢复
  8. 工作台式计算机配置单,台式电脑配置清单.doc
  9. 多关键词采集搜索引擎URL网址域名
  10. android 找不到手机,找不到 Android 开发者选项,难道我的手机系统没有?| 有轻功 #290...
  11. C#轻松创建ModbusTCP服务器【Slave】,实现工业数据交换接口。
  12. MATLAB学习笔记(二)——使用GUIDE构建App
  13. 如何使用 ABAP 代码发送带有 PDF 附件的电子邮件
  14. 斩获技术向善奖,云开发上榜 2020 中国技术品牌影响力企业
  15. 不要在该约炮的年纪谈佛系
  16. 657UVa掷骰子(图)
  17. CANoe——CAPL案例
  18. 利用matlab对纯电动汽车在nedc工况下的燃油经济性进行仿真
  19. mysql之字符串连接操作
  20. VisionPro工具小结

热门文章

  1. 计算机原理与基础 —— 原码、反码、补码之间的快速转换和简单运算
  2. [Ynoi2019模拟赛]Yuno loves sqrt technology II
  3. JVM内存GC的骗局
  4. RT ROM boot简介
  5. 多个反斜杠的消除处理
  6. 初学spring mvc
  7. JDK的环境变量配置
  8. 猪行天下之Python基础——10.1 Python常用模块(上)
  9. 小龙虾上天了!口碑APP推出飞机上扫码点餐服务
  10. Mysql中where条件一个单引号引发的性能损耗