寻求对Function <T,R>的深入理解

时间:2019-10-19 23:23:50

标签: java functional-interface functional-java

下面显示的是一些使用Java Streams的示例代码。我的问题特别涉及Interface Function<T,R>,它接受​​类型为T的输入并返回类型为R的东西。

import static java.util.Arrays.asList;
import static java.util.stream.Collectors.groupingBy;

import java.util.List;
import java.util.Map;

public class Dish {

  private final String name;
  private final boolean vegetarian;
  private final String calories;

  public Dish(String name, boolean vegetarian, String calories) {
    this.name = name;
    this.vegetarian = vegetarian;
    this.calories = calories;
  }

  public String getName() {
    return name;
  }

  public boolean isVegetarian() {
    return vegetarian;
  }

  public String getCalories() {
    return calories;
  }

  @Override
  public String toString() {
    return name;
  }

  public static final List<Dish> menu = asList(
      new Dish("pork", false, "GE 600"),
      new Dish("beef", false, "GE 600"),
      new Dish("chicken", false, "300-600"),
      new Dish("french fries", true, "300-600"),
      new Dish("rice", true, "LE 300"),
      new Dish("season fruit", true, "LE 300"),
      new Dish("pizza", true, "300-600"),
      new Dish("prawns", false, "300-600"),
      new Dish("salmon", false, "300-600")
  );

  public static void main(String[] args) {
        Map<String, List<Dish>> dishByCalories = menu.stream()
                .collect(groupingBy(Dish::getCalories));
        System.out.println(dishByCalories);
  }
}

很明显groupingBy(Dish::getCalories)符合collect(即Collector<? super T,A,R> collector)的预期方法签名要求

现在进入groupingsBy,其签名要求如下:
static <T,K> Collector<T,?,Map<K,List<T>>> groupingBy(Function<? super T,? extends K> classifier)

我们传递给groupingsBy的方法引用是Dish::getCalories

很显然,Dish::getCalories满足Function<? super T,? extends K>的签名要求(即它应该接受T的某些超类的输入并返回K的某个子类的结果)。

但是,getCalories方法不接受任何参数,它返回一个字符串。

请帮助消除我的困惑。

1 个答案:

答案 0 :(得分:5)

getCalories instance 方法,因此它接受隐式this自变量。

Dish::getCalories等同于lambda (Dish dish) -> dish.getCalories(),这使之成为Function<Dish, String>的实现。