详解JAVA中的for-each循环与迭代

作者:知其然,后知其所以然 时间:2022-05-11 12:29:14 

在学习java中的collection时注意到,collection层次的根接口Collection实现了Iterable<T>接口(位于java.lang包中),实现这个接口允许对象成为 "foreach" 语句的目标,而此接口中的唯一方法,实现的就是返回一个在一组 T 类型的元素上进行迭代的迭代器。

一、迭代器Iterator

接口:Iterator<T>


public interface Iterator<E>{
 boolean hasNext();
E next();
void remove();
}

查看Iterator接口API可以知道,这是对collection进行迭代的迭代器。迭代器允许调用者利用定义良好的语义在迭代期间从迭代器所指向的 collection 移除元素。

尤其值得注意的是此迭代器remove()方法的使用:从迭代器指向的 collection 中移除迭代器返回的最后一个元素(可选操作)。每次调用 next 只能调用一次此方法。如果进行迭代时用调用此方法(remove方法)之外的其他方式修改了该迭代器所指向的 collection,则迭代器的行为是不确定的。 接口设计人员在设计Iterator<T>接口的时候已经指出,在进行迭代时如果调用了除了迭代器的remove()方法修改了该迭代器所指向的collection,则会造成不确定的后果。具体出现什么后果依迭代器的具体实现而定。针对这种不确定的后果可能出现的情况,在学习ArrayList时遇到了其中一种:迭代器抛出 ConcurrentModificationException异常。具体异常情况如下代码所示:


import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;

public class ItaratorTest {

public static void main(String[] args) {
   Collection<String> list = new ArrayList<String>();
   list.add("Android");
   list.add("IOS");
   list.add("Windows Mobile");

Iterator<String> iterator = list.iterator();
   while (iterator.hasNext()) {
     String lang = iterator.next();
     list.remove(lang);//will throw ConcurrentModificationException
   }
 }

}

此段代码在运行时会抛出ConcurrentModificationException异常,因为我们在迭代器运行期间没有用iterator的remove()方法来删除元素,而是使用ArrayList的 remove()方法改变了迭代器所指向的collection。这就违反了迭代器的设计原则,所以发生了异常。

所报异常情况如下所示:

Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:859)
    at java.util.ArrayList$Itr.next(ArrayList.java:831)
    at Text.ItaratorTest.main(ItaratorTest.java:17)

二、for-each循环与迭代器Iterator<T>

从Java5起,在Java中有了for-each循环,可以用来循环遍历collection和array。Foreach循环允许你在无需保持传统for循环中的索引,或在使用iterator /ListIterator(ArrayList中的一种迭代器实现)时无需调用while循环中的hasNext()方法就能遍历collection。for-each循环简化了任何Collection或array的遍历过程。但是使用foreach循环也有两点需要注意。

使用foreach循环的对象,必须实现了Iterable<T>接口

请看如下示例:


import java.util.ArrayList;

public class ForeachTest1 {

public static void main(String args[]) {
   CustomCollection<String> myCollection = new CustomCollection<String>();
   myCollection.add("Java");
   myCollection.add("Scala");
   myCollection.add("Groovy");

// What does this code will do, print language, throw exception or
   // compile time error
   for (String language : myCollection) {
     System.out.println(language);
   }
 }

private class CustomCollection<T> {
   private ArrayList<T> bucket;

public CustomCollection() {
     bucket = new ArrayList();
   }

public int size() {
     return bucket.size();
   }

public boolean isEmpty() {
     return bucket.isEmpty();
   }

public boolean contains(T o) {
     return bucket.contains(o);
   }

public boolean add(T e) {
     return bucket.add(e);
   }

public boolean remove(T o) {
     return bucket.remove(o);
   }

}
}

上述代码将无法通过编译,这是因为代码中的CustomCollection类没有实现Iterable<T>接口,编译期的报错如下:

Exception in thread "main" java.lang.Error: Unresolved compilation problem:
    Can only iterate over an array or an instance of java.lang.Iterable

    at Text.ForeachTest1.main(ForeachTest1.java:15)

事实上,无需等到编译时才发现报错,eclipse会在这段代码写完之后就会在foreach循环处显示错误:Can only iterate over an array or an instance of java.lang.Iterable

从上述示例可以再次得到确认的是,foreach循环只适用于实现了Iterable<T>接口的对象。由于所有内置Collection类都实现了java.util.Collection接口,已经继承了Iterable,所以为了解决上述问题,可以选择简单地让CustomCollection实现Collection接口或者继承AbstractCollection。解决方式如下:


import java.util.AbstractCollection;
import java.util.ArrayList;
import java.util.Iterator;

public class ForeachTest {
 public static void main(String args[]) {
   CustomCollection<String> myCollection = new CustomCollection<String>();
   myCollection.add("Java");
   myCollection.add("Scala");
   myCollection.add("Groovy");
   for (String language : myCollection) {
     System.out.println(language);
   }
 }

private static class CustomCollection<T> extends AbstractCollection<T> {
   private ArrayList<T> bucket;

public CustomCollection() {
     bucket = new ArrayList();
   }

public int size() {
     return bucket.size();
   }

public boolean isEmpty() {
     return bucket.isEmpty();
   }

public boolean contains(Object o) {
     return bucket.contains(o);
   }

public boolean add(T e) {
     return bucket.add(e);
   }

public boolean remove(Object o) {
     return bucket.remove(o);
   }

@Override
   public Iterator<T> iterator() {
     // TODO Auto-generated method stub
     return bucket.iterator();
   }
 }
}

2.foreach循环的内部实现也是依靠Iterator进行实现的

为了验证foreach循环是使用Iterator作为内部实现这一事实,我们依然采用本文最开始的实例进行验证:


public class ItaratorTest {

public static void main(String[] args) {
   Collection<String> list = new ArrayList<String>();
   list.add("Android");
   list.add("IOS");
   list.add("Windows Mobile");

// example1
   // Iterator<String> iterator = list.iterator();
   // while (iterator.hasNext()) {
   // String lang = iterator.next();
   // list.remove(lang);
   // }

// example 2
   for (String language : list) {
     list.remove(language);
   }
 }

}

程序运行时所报异常:

Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:859)
    at java.util.ArrayList$Itr.next(ArrayList.java:831)
    at Text.ItaratorTest.main(ItaratorTest.java:22)

此异常正说明了for-each循环内部使用了Iterator来遍历Collection,它也调用了Iterator.next(),这会检查(元素的)变化并抛出ConcurrentModificationException。

总结:

  • 在遍历collection时,如果要在遍历期间修改collection,则必须通过Iterator/listIterator来实现,否则可能会发生“不确定的后果”。

  • foreach循环通过iterator实现,使用foreach循环的对象必须实现Iterable接口

标签:java,for-each
0
投稿

猜你喜欢

  • springboot使用redisRepository和redistemplate操作redis的过程解析

    2023-10-11 06:57:03
  • Java并发包线程池ThreadPoolExecutor的实现

    2022-11-10 09:52:41
  • 解决java.lang.Error: Unresolved compilation problems:问题

    2023-02-10 05:58:08
  • C#实现redis读写的方法

    2023-07-13 16:21:35
  • C#从DataTable获取数据的方法

    2022-09-16 03:20:59
  • java设计模式之工厂模式实例详解

    2023-11-24 23:35:12
  • C#在图片增加文字的实现代码

    2023-03-30 03:26:24
  • Java封装、继承、多态三大特征的理解

    2023-07-16 14:26:46
  • Struts2中接收表单数据的三种驱动方式

    2022-04-21 09:23:11
  • 利用Kotlin开发你的第一个Android应用

    2022-04-23 14:39:53
  • Java 10 局部变量类型推断浅析

    2023-11-25 06:24:13
  • C#创建一个Word并打开的方法

    2022-08-04 23:33:13
  • springboot整合mybatis实现数据库的更新批处理方式

    2023-11-29 07:08:37
  • java中使用zxing批量生成二维码立牌

    2021-12-31 04:31:03
  • IDEA SpringBoot项目配置热更新的步骤详解(无需每次手动重启服务器)

    2023-11-12 00:22:41
  • java设计模式之工厂方法模式

    2022-08-29 05:14:10
  • Activiti7整合Springboot使用记录

    2022-11-11 06:17:24
  • C# 获取硬件参数的实现方法

    2023-11-04 21:30:38
  • C# MVC 微信支付教程系列之公众号支付代码

    2022-11-29 09:14:29
  • 手把手教你JAVA进制之间的转换

    2023-11-14 23:18:56
  • asp之家 软件编程 m.aspxhome.com