什么是迭代器(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)
评论()
编辑
收藏
举报