<返回更多

Java中不建议使用foreach的六大场景

2023-12-07  今日头条  迷路的架构师
加入收藏

JAVA中,foreach 是一个常用的循环结构,它可以极大地简化遍历数组或集合(例如 List 或 Set)的代码。它通常被认为是一种更加简洁和易读的迭代方式。然而,可能有一些情况下不建议使用 foreach 循环:

  1. 移除元素: 使用 foreach 循环时,如果尝试直接从正在遍历的集合中移除元素,可能会抛出 ConcurrentModificationException。这是因为 foreach 循环背后使用的是迭代器,而直接修改集合会导致迭代器的状态与实际的集合状态不一致。在这种情况下,你应该使用显式迭代器并调用 iterator.remove() 方法。
// 使用迭代器来安全地移除集合中的元素:

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

public class RemoveElement {
    public static void mAIn(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("A");
        list.add("B");
        list.add("C");

        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()) {
            String item = iterator.next();
            if (item.equals("B")) {
                iterator.remove(); // 安全移除元素
            }
        }

        System.out.println(list); // 输出结果将不包含"B"
    }
}
  1. 性能敏感: 如果你正在处理超大数据集,或者在性能要求非常严格的场景中,foreach 循环可能会引入轻微的性能开销,因为它需要构造一个迭代器对象。对于原始类型的数组,使用传统的 for 循环可以避免自动装箱和拆箱的额外开销,并提供更好的性能。
// 使用传统的for循环处理原始类型数组:
public class PerformanceSensitive {
    public static void main(String[] args) {
        int[] numbers = {1, 2, 3, 4, 5};

        // 使用传统 for 循环来避免可能的性能开销
        for (int i = 0; i < numbers.length; i++) {
            System.out.println(numbers[i]);
        }
    }
}
  1. 需要修改当前元素: 在 foreach 循环中,没有直接的方式来修改当前遍历到的元素,因为所得到的只是一个副本。如果你需要修改列表中的元素,你通常需要使用传统的 for 循环,以便获得元素的索引。
// 通过传统的for循环获取索引并修改数组或列表中的元素:
import java.util.ArrayList;
import java.util.List;

public class ModifyElement {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("Apple");
        list.add("banana");
        list.add("cherry");

        for (int i = 0; i < list.size(); i++) {
            list.set(i, list.get(i).toUpperCase());
        }

        System.out.println(list); // 所有元素变为大写
    }
}
  1. 需要索引或迭代器: foreach 循环不提供当前元素的索引或迭代器本身。如果你的逻辑需要使用元素的索引,或者你需要在迭代过程中获取迭代器来执行其他操作,你应该使用传统的 for 循环或者直接使用迭代器。
// 使用传统的for循环以获取元素的索引:
import java.util.ArrayList;
import java.util.List;

public class NeedIndex {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("one");
        list.add("two");
        list.add("three");

        for (int i = 0; i < list.size(); i++) {
            System.out.println("Index " + i + ": " + list.get(i));
        }
    }
}
  1. 多个集合并行遍历: 如果你需要同时遍历两个集合,并且它们是相关联的,例如键值对的情况下,使用 foreach 循环可能会变得复杂。在这种情况下,使用传统的 for 循环通常更方便,因为你可以控制多个索引或迭代器。
// 假设有两相关联的集合,一个是键的列表 keys,另一个是值的列表 values
List<String> keys = Arrays.asList("key1", "key2", "key3");
List<String> values = Arrays.asList("value1", "value2", "value3");

// 使用传统的 for 循环同时遍历 keys 和 values 集合
for (int i = 0; i < keys.size() && i < values.size(); i++) {
    String key = keys.get(i);
    String value = values.get(i);
    System.out.println(key + ": " + value);
}
  1. 特定的并发集合: 当使用特定的线程安全集合类,如 CopyOnWriteArrayList 时,foreach 循环由于内部持有整个迭代期间的集合快照,可能会导致预期之外的内存消耗。
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;

public class ForeachCopyOnWriteExample {
    public static void main(String[] args) {
        // 使用 CopyOnWriteArrayList 创建线程安全的 ArrayList
        List<String> list = new CopyOnWriteArrayList<>();
        list.add("Element1");
        list.add("Element2");
        list.add("Element3");

        // 使用 foreach 循环遍历 CopyOnWriteArrayList
        for (String element : list) {
            System.out.println(element);
            // 此处修改集合内容不会影响迭代,因为使用的是集合快照
            list.add("ElementNew");
        }

        // 最后打印集合的内容,可以看到新元素已经被添加
        System.out.println("After modifications:");
        for (String element : list) {
            System.out.println(element);
        }
    }
}

CopyOnWriteArrayList 类创建了一个线程安全的集合。当我们在 foreach 循环中遍历集合并同时向其中添加新元素时,由于 CopyOnWriteArrayList 内部实现了对原始集合的复制(即创建了快照),foreach 循环使用的是开始迭代时的集合状态,所以迭代过程中集合状态的改变不会影响到迭代本身。这可能导致大量内存的额外消耗,尤其是当集合很大时。

关键词:Java      点击(8)
声明:本站部分内容来自互联网,如有版权侵犯或其他问题请与我们联系,我们将立即删除或处理。
▍相关推荐
更多Java相关>>>