集合输出实际上从JDK1.8开始就在Iterable接口中提供了一个forEach()方法,但是这种方法输出并不是传统意义上集合输出形式,并且也很难在实际的开发之中出现,对于集合操作而言,一共有四种输出形式:Iterator迭代输出(95%)、ListIterator双向迭代输出(0.1%)、Enumeration枚举输出(4.9%)、foreach输出(与Iterator相当)。
Iterator迭代输出
通过Collection接口的继承关系可以发现,从JDK1.5开始其多继承了一个Iterable父接口,并且在这个接口里面定义有一个iterator()操作方法,通过此方法可以获取Iterator接口对象(在JDK1.5之前,这一方法直接定义在Collection接口之中)。
获取Iterator接口对象:public Iterator iterator();
在Iterator接口里面定义有如下的方法:
No. | 方法名称 | 类型 | 描述 |
---|---|---|---|
01 | public boolean hasNext() | 普通 | 判断是否有数据 |
02 | public E next() | 普通 | 取出当前数据 |
03 | public default void remove() | 普通 | 删除 |
在之前使用的java.util.Scanner类就是Iterator接口的子类,所以此时类继承关系如下:
范例:使用Iterator输出
package cn.mldn.demo;
import java.util.Set;
import java.util.Iterator;
public class JavaAPIDemo {
public static void main(String[] args) throws Exception {
Set<String> all = Set.of("Hello", "World", "MLDN");
Iterator<String> iter = all.iterator(); //实例化Iterator接口对象
while (iter.hasNext()) {
String str = iter.next();
System.out.println(str); // World Hello MLDN
}
}
}
但是对于Iterator接口中的remove()方法的使用需要特别注意一下(如果不是必须不要使用)。实际上在Collection接口中定义有数据的删除操作方法,但是在进行迭代输出的过程中如果你使用了Collection中的remove()方法会导致迭代失败。
范例:采用Collection集合中remove()方法删除
package cn.mldn.demo;
import java.util.Set;
import java.util.Iterator;
import java.util.HashSet;
public class JavaAPIDemo {
public static void main(String[] args) throws Exception {
Set<String> all = new HashSet<String>();
all.add("Hello");
all.add("World");
all.add("MLDN");
Iterator<String> iter = all.iterator(); //实例化Iterator接口对象
while (iter.hasNext()) {
String str = iter.next();
if ("World".equals(str)) {
all.remove("World"); //Collection集合方法
}else {
System.out.println(str); // Hello Exception in thread "main" java.util.ConcurrentModificationException
}
}
}
}
程序运行结果:
Hello
Exception in thread "main" java.util.ConcurrentModificationException
此时无法进行数据删除处理操作,那么就只能够利用Iterator接口中的remove()方法删除。
范例:使用Iterator接口删除方法
package cn.mldn.demo;
import java.util.Set;
import java.util.Iterator;
import java.util.HashSet;
public class JavaAPIDemo {
public static void main(String[] args) throws Exception {
Set<String> all = new HashSet<String>();
all.add("Hello");
all.add("World");
all.add("MLDN");
Iterator<String> iter = all.iterator(); //实例化Iterator接口对象
while (iter.hasNext()) {
String str = iter.next();
if ("World".equals(str)) {
iter.remove(); //删除当前的数据
}else {
System.out.println(str); // Hello Exception in thread "main" java.util.ConcurrentModificationException
}
}
System.out.println("*** "+ all);
}
}
//Hello
//MLDN
//*** [Hello, MLDN]
此时程序执行后没有出现任何的错误,并且可以成功的删除原始集合中的数据。
面试题:
请解释Collection.remove()与Iterator.remove()的区别?
在进行迭代输出的时候,如果使用了Collection.remove()则会造成并发更新的异常,导致程序删除出错,而此时只能够利用Iterator接口中remove()方法实现正常的删除处理。