本文由 JavaGuide 翻译,原文地址:https://www.baeldung.com/foreach-java

1 概述

在Java 8中引入的forEach循环为程序员提供了一种新的,简洁而有趣的迭代集合的方式。

在本文中,我们将看到如何将forEach与集合一起使用,它采用何种参数以及此循环与增强的for循环的不同之处。

2 基础知识

  1. public interface Collection<E> extends Iterable<E>

Collection 接口实现了 Iterable 接口,而 Iterable 接口在 Java 8开始具有一个新的 API:

  1. void forEach(Consumer<? super T> action)//对 Iterable的每个元素执行给定的操作,直到所有元素都被处理或动作引发异常。

使用forEach,我们可以迭代一个集合并对每个元素执行给定的操作,就像任何其他迭代器一样。

例如,迭代和打印字符串集合_的_for循环版本:

  1. for (String name : names) {
  2. System.out.println(name);
  3. }

我们可以使用forEach写这个 :

  1. names.forEach(name -> {
  2. System.out.println(name);
  3. });

3.使用forEach方法

3.1 匿名类

我们使用 forEach迭代集合并对每个元素执行特定操作。要执行的操作包含在实现Consumer接口的类中,并作为参数传递给forEach 。

所述消费者接口是一个功能接口(具有单个抽象方法的接口)。它接受输入并且不返回任何结果。

Consumer 接口定义如下:

  1. @FunctionalInterface
  2. public interface Consumer {
  3. void accept(T t);
  4. }

任何实现,例如,只是打印字符串的消费者:

  1. Consumer<String> printConsumer = new Consumer<String>() {
  2. public void accept(String name) {
  3. System.out.println(name);
  4. };
  5. };

可以作为参数传递给forEach

  1. names.forEach(printConsumer);

但这不是通过消费者和使用forEach API 创建操作的唯一方法。让我们看看我们将使用forEach方法的另外2种最流行的方式:

3.2 Lambda表达式

Java 8功能接口的主要优点是我们可以使用Lambda表达式来实例化它们,并避免使用庞大的匿名类实现。

由于 Consumer 接口属于函数式接口,我们可以通过以下形式在Lambda中表达它:

  1. (argument) -> { body }
  2. name -> System.out.println(name)
  3. names.forEach(name -> System.out.println(name));

3.3 方法参考

我们可以使用方法引用语法而不是普通的Lambda语法,其中已存在一个方法来对类执行操作:

  1. names.forEach(System.out::println);

4.forEach在集合中的使用

4.1.迭代集合

任何类型Collection的可迭代 - 列表,集合,队列 等都具有使用forEach的相同语法。

因此,正如我们已经看到的,迭代列表的元素:

  1. List<String> names = Arrays.asList("Larry", "Steve", "James");
  2. names.forEach(System.out::println);

同样对于一组:

  1. Set<String> uniqueNames = new HashSet<>(Arrays.asList("Larry", "Steve", "James"));
  2. uniqueNames.forEach(System.out::println);

或者让我们说一个队列也是一个集合

  1. Queue<String> namesQueue = new ArrayDeque<>(Arrays.asList("Larry", "Steve", "James"));
  2. namesQueue.forEach(System.out::println);

4.2.迭代Map - 使用Map的forEach

Map没有实现Iterable接口,但它提供了自己的forEach 变体,它接受BiConsumer。*

  1. Map<Integer, String> namesMap = new HashMap<>();
  2. namesMap.put(1, "Larry");
  3. namesMap.put(2, "Steve");
  4. namesMap.put(3, "James");
  5. namesMap.forEach((key, value) -> System.out.println(key + " " + value));

4.3.迭代一个Map - 通过迭代entrySet

  1. namesMap.entrySet().forEach(entry -> System.out.println(entry.getKey() + " " + entry.getValue()));