目录
1、条件差别
2、参数类型差别
3、用法差别
4、代码分析
4.1 for循环
4.2 forEach
4.3 iterator遍历
1、条件差别
- for需要知道集合或数组的大小,而且需要是有序的,不然无法遍历;
- foreach和iterator都不需要知道集合或数组的大小,他们都是得到集合内的每个元素然后进行处理;
2、参数类型差别
- for和foreach都需要先知道集合的类型,甚至是集合内元素的类型,即需要访问内部的成员,不能实现态;
- iterator是一个接口类型,他不关心集合或者数组的类型,而且他还能随时修改和删除集合的元素。
当我们需要遍历不同的集合时,我们只需要传递集合的iterator。这就是iterator的好处,他不包含任何有关他所遍历的序列的类型信息,能够将遍历序列的操作与序列底层的结构分离。
迭代器统一了对容器的访问方式。这也是接口的解耦的最好体现。
3、用法差别
- for循环一般用来处理比较简单的有序的,可预知大小的集合或数组.
- foreach可用于遍历任何集合或数组,而且操作简单易懂,他唯一的不好就是需要了解集合内部类型.
- iterator是最强大的,他可以随时修改或者删除集合内部的元素,并且是在不需要知道元素和集合的类型的情况下进行的,当你需要对不同的容器实现同样的遍历方式时,迭代器是最好的选择!
4、代码分析
4.1 for循环
public static void main(String[] args) throws FileNotFoundException {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
for(int i = 0; i < list.size(); i++){
if(list.get(i).equals(3)) list.remove(i);
}
System.out.println(list);
}
打印结果:
4.2 forEach
public static void main(String[] args) throws FileNotFoundException {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
for(Integer i : list){
if(i.equals(3)) list.remove(i);
}
System.out.println(list);
}
打印结果:
4.3 iterator遍历
public static void main(String[] args) throws FileNotFoundException {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
Iterator<Integer> iterator = list.iterator();
while(iterator.hasNext()){
Integer i = iterator.next();
if(i.equals(3)) list.remove(i);
}
System.out.println(list);
}
打印结果:
原因 : 在集合增加同时,又做了删除操作,导致了list集合中 modCount 和 expectedModCount 不一致,导致抛出异常。这里用的是 list并且是单线程,已导致了异常报错,多线程情况下,异常报错的几率更高。
可以使用CopyOnWriteArrayList,Collections.synchronizedCollection()等并发容器,来解决此类问题。