迭代器

什么是迭代器(Iterator)?

  • Iterator接口提供了很多对集合元素进行迭代的方法,每一个集合类都包含了可以返回迭代器实例的迭代方法。
  • 迭代器可以在迭代的过程中删除底层集合的元素,但是不可以直接调用集合的remove(Object Obj)删除,可以通过迭代器的remove()方法删除。

Iterator和ListIterator的区别是什么?

  • Iterator可用来遍历Set和List集合,但是ListIterator只能用来遍历List。
  • Iterator对集合只能是前向遍历,ListIterator既可以前向也可以后向。
  • ListIterator实现了Iterator接口,并包含其他的功能,比如:增加元素,替换元素,获取前一个和后一个元素的索引,等等。

快速失败(fail-fast)和安全失败(fail-safe)的区别是什么?

一:快速失败(fail—fast)
  • 在用迭代器遍历一个集合对象时,如果遍历过程中对集合对象的结构进行了修改(增加、删除),则会抛出Concurrent Modification Exception。
原理:
  • 迭代器在遍历时直接访问集合中的内容,并且在遍历过程中使用一个 modCount 变量。
  • 集合在被遍历期间如果结构发生变化,就会改变modCount的值。
  • 每当迭代器使用hashNext()/next()遍历下一个元素之前,都会检测modCount变量是否为expectedmodCount值,是的话就返回遍历;否则抛出异常,终止遍历。
注意:
  • 这里异常的抛出条件是检测到 modCount!=expectedmodCount 这个条件。
  • 如果集合发生变化时修改modCount值刚好又设置为了expectedmodCount值,则异常不会抛出。
  • 因此,不能依赖于这个异常是否抛出而进行并发操作的编程,这个异常只建议用于检测并发修改的bug。
场景:
  • java.util包下的集合类都是快速失败的,不能在多线程下发生并发修改(迭代过程中被修改)。
二:安全失败(fail—safe)
  • 采用安全失败机制的集合容器,在遍历时不是直接在集合内容上访问的,而是先复制原有集合内容,在拷贝的集合上进行遍历。
原理:
  • 由于迭代时是对原集合的拷贝进行遍历,所以在遍历过程中对原集合所作的修改并不能被迭代器检测到,所以不会触发Concurrent Modification Exception。
缺点:
  • 基于拷贝内容的优点是避免了Concurrent Modification Exception。
  • 但同样地,迭代器并不能访问到修改后的内容,即:迭代器遍历的是开始遍历那一刻拿到的集合拷贝,在遍历期间原集合发生的修改迭代器是不知道的。
场景:
  • java.util.concurrent包下的容器都是安全失败,可以在多线程下并发使用,并发修改。

Enumeration接口和Iterator接口的区别有哪些?

  • Enumeration速度是Iterator的2倍,同时占用更少的内存。
  • 但是,Iterator远远比Enumeration安全,因为其他线程不能够修改正在被iterator遍历的集合里面的对象。
  • 同时,Iterator允许调用者删除底层集合里面的元素,这对Enumeration来说是不可能的。
posted @ 2021-01-12 13:27  张三丰学Java  阅读(88)  评论(0编辑  收藏  举报