Java中的Func编程.Method-ref与功能应用签名不匹配但成功运行

tez*_*zza 3 java functional-programming java-8 method-reference

我目前正在阅读Venkat Subramaniam的Java函数编程,到目前为止,它是一本很棒的读物.

然而,有一个特殊的例子让我感到困惑.在下面的代码示例中,有一个名为的方法setFilters(final Function<Color,Color>... filters).

这需要一个实现Function接口的列表.在代码setFilters使用方法引用叫...... camera.setFilters(Color::lighter, Color::darker);.

这很令人困惑,因为Color类没有实现Function接口,也没有匹配R apply(T t);签名的方法.

我错过了一些明显的东西吗?有人Color::darker可以解释如何将etc传递给期望的参数Function<Color,Color>

这是完整的代码示例:

package designing.fpij;

import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;

import java.util.stream.Stream;
import java.util.function.Function;
import java.awt.Color;
import java.util.function.Consumer;

@SuppressWarnings("unchecked")
public class Camera {  
  private Function<Color, Color> filter;


  public Color capture(final Color inputColor) {
    final Color processedColor = filter.apply(inputColor);
    return processedColor;
  }

  public void setFilters(final Function<Color, Color>... filters) {
    filter =
            Stream.of(filters).
                    reduce((after, before) -> filter.compose(before)).orElse(color -> color);
  }
  public Camera() {
    setFilters();
  }

  public static void main(final String[] args) {
    final Camera camera = new Camera();
    final Consumer<String> printCaptured = (filterInfo) ->
            System.out.println(String.format("with %s: %s", filterInfo, camera.capture(new Color(200, 100, 150))));

    System.out.println("//" + "START:NOFILTER_OUTPUT");
    printCaptured.accept("no filter");
    System.out.println("//" + "END:NOFILTER_OUTPUT");

    System.out.println("//" + "START:BOTH_OUTPUT");
    camera.setFilters(Color::brighter, Color::darker);
    printCaptured.accept("brighter & darker filter");
    System.out.println("//" + "END:BOTH_OUTPUT");

  }

}
Run Code Online (Sandbox Code Playgroud)

Tri*_*sha 5

调用

camera.setFilters(Color::darker);
Run Code Online (Sandbox Code Playgroud)

相当于

setFilters(new Function<Color, Color>() {
    @Override
    public Color apply(Color color) {
        return color.darker();
    }
});
Run Code Online (Sandbox Code Playgroud)

(如果您使用的是Java-8之前的匿名内部类).您可以看到添加Color不需要的所有样板文件FunctionalInterface或实现Function- 在更长的形式中,您将在a中包装对此方法的调用Function.

简化对lambda的这个调用给出了

setFilters((Function<Color, Color>) color -> color.darker());
Run Code Online (Sandbox Code Playgroud)

甚至更短:

setFilters(color -> color.darker());
Run Code Online (Sandbox Code Playgroud)

最终可以简化为您的方法参考:

setFilters(Color::darker);
Run Code Online (Sandbox Code Playgroud)

  • 是的,我使用IntelliJ IDEA为我执行此操作 - 在方法引用上按"Alt + Enter"可以选择扩展为lambda,然后可以使用相同的过程扩展为匿名内部类 (2认同)