foreach是java5的新特性,增强了for循环,在遍历数组和集合上面很方便。
foreach是for循环的简化版本,但是foreach不能完全取代for循环,然而任何foreach都可以改写为for循环,foreach不是一个关键字,主要就是这个格式,每一个for的意思。
foreach语句格式:
for(元素类型type 元素变量value : 遍历对象obj) {
引用x的java语句;
}
foreach使用场景
* 遍历数组
遍历数组时,foreach转换为对数组中的每一个元素的循环引用,相当于for语法循环遍历一样
int arr[][] = {{1,2},{3,4}};
for(int[] x:arr) {
for(int e:x) {
System.out.print(e+ " ");//逐个输出数组元素,多维数组的本质就是没有多维数组
}
System.out.println();
}
* 遍历实现了Iterable接口的对象(如集合)
对有实现Iterable接口的对象采用foreach语法糖的话,编译器会调用Iterable接口的 iterator方法来遍历对象,iterator方法中是调用Iterator接口的的 next()和hasNext()方法来做遍历。JAVA中有一个叫做迭代器模式的设计模式,这个其实就是对迭代器模式的一个实现
foreach使用限制
在用foreach循环遍历一个集合时不能向集合中增加元素,不能从集合中删除元素,否则会抛出ConcurrentModificationException异常。抛出该异常是因为在集合内部有一个modCount变量用于记录集合中元素的个数,当向集合中增加或删除元素时,modCount也会随之变化,在遍历开始时会记录modCount的值,每次遍历元素时都会判断该变量是否发生了变化,如果发生了变化则抛出ConcurrentModificationException异常
List list = new ArrayList();
list.add("0");
list.add("1");
list.add("2");
for(String str:list){
System.out.println(str);
list.add("3");
}
当使用foreach循环基本类型时变量时不能修改集合中的元素的值;遍历对象时可以修改对象的属性的值,但是不能修改对象的引用(因为foreach获得的是集合中变量的一个副本)
当在遍历时需要根据元素在集合中的index的时候不能使用foreach
for与foreach效率比较
实现RandomAccess接口的类实例,假如是随机访问的,使用普通for循环效率将高于使用foreach循环;反过来,如果是顺序访问的,则使用Iterator会效率更高。
比如,对于实现RandomAccess接口的ArrayList,通过随机访问方式使用for循环遍历ArrayList比利用顺序访问方式的foreach循环遍历效率高一些,而对于LinkList,在无法利用随机访问方式遍历的情况下,使用foreach循环效率更高一些。
public static void main(String[] args){
//实例化arrayList
List arrayList = new ArrayList();
//实例化linkList
List linkList = new LinkedList();
//插入10万条数据
for (int i = 0; i < 100000; i++) {
arrayList.add(i);
linkList.add(i);
}
int array = 0;
//用for循环arrayList
long arrayForStartTime = System.currentTimeMillis();
for (int i = 0; i < arrayList.size(); i++) {
array = arrayList.get(i);
}
long arrayForEndTime = System.currentTimeMillis();
System.out.println("用for循环arrayList 10万次花费时间:" + (arrayForEndTime - arrayForStartTime) + "毫秒");
//用foreach循环arrayList
long arrayForeachStartTime = System.currentTimeMillis();
for(Integer in : arrayList){
array = in;
}
long arrayForeachEndTime = System.currentTimeMillis();
System.out.println("用foreach循环arrayList 10万次花费时间:" + (arrayForeachEndTime - arrayForeachStartTime ) + "毫秒");
//用for循环linkList
long linkForStartTime = System.currentTimeMillis();
int link = 0;
for (int i = 0; i < linkList.size(); i++) {
link = linkList.get(i);
}
long linkForEndTime = System.currentTimeMillis();
System.out.println("用for循环linkList 10万次花费时间:");
System.out.println("用for循环linkList 10万次花费时间:" + (linkForEndTime - linkForStartTime) + "毫秒");
//用froeach循环linkList
long linkForeachStartTime = System.currentTimeMillis();
for(Integer in : linkList){
link = in;
}
long linkForeachEndTime = System.currentTimeMillis();
System.out.println("用foreach循环linkList 10万次花费时间:" + (linkForeachEndTime - linkForeachStartTime ) + "毫秒");
}
结论:
循环遍历数组结构的数据时,建议使用普通for循环,因为for循环采用下标访问,对于数组结构的数据来说,采用下标随机访问效率较高。
循环遍历链表结构的数据时,一定不要使用普通for循环,这种做法很糟糕,数据量大的时候有可能会导致系统崩溃。