函数接口是Java 8最重要的概念之一,实际上为lambda表达式提供了动力,但是许多开发人员没有首先了解函数接口在Java 8中的作用就花了很多精力来理解它,并花时间学习lambda表达式和Stream API。除非您知道什么是功能接口以及lambda与它之间的关系,否则您将无法使用Java 8的强大功能,例如lambda表达式和流API 。 没有功能接口的知识,您将无法理解在代码中可以使用lambda的位置,但是您将很难编写该方法所期望的lambda表达式,因此,对功能接口有一个很好的了解非常重要。 Java 8。
在本文中,我将通过解释什么是函数接口,什么是@Functional
注释,它们与lambda表达式之间的关系以及它们如何帮助您在代码中使用lambda表达式来填补这一空白。 那么,让我们从第一件事开始,什么是功能接口?
Java 8中的功能接口是什么
好的,功能接口不过是带有一个抽象方法的接口,例如Comparable
, Runnable
, EventListener
, Comparator
等。您可以看到这些接口甚至在JDK 8之前就已经存在于Java中,但是为什么我们将这种接口称为功能接口呢?
这是一个很好的问题,如果您对函数式编程有所了解,就知道它可以传递代码,即函数,就像将数据或对象传递给方法一样。
这些接口只有一种抽象方法被用于传递代码,就像您以函数式编程语言传递函数一样, 这就是为什么它们被称为函数接口 。
例如,您可以通过实现Comparator
接口来创建一个Anonymous类,从而直接传递代码以比较对象,如下所示:
Collections.sort(list, new Comparator(){public int compare(String s1, String s2){return s1.length() - s2.length();}});
因此,如果仔细观察,您会发现我们正在使用这些接口将代码传递给函数 。 它们也称为策略接口,因为这是策略模式的实现,其中,构成策略的代码被注入到在运行时运行该策略的代码中。
顺便说一句,如果您不知道什么是策略模式,那么我建议您从0转到1:设计模式–至关重要的24 –在Java中 ,因为设计模式的知识对于Java的有效编码很重要。
那么,既然我们知道什么是功能接口,让我们了解它们与lambda表达式之间的关系 ,以及对功能接口的理解对于使用lambda表达式编写代码有何重要性?
好吧,要记住的最重要的事情是,Java中lambda表达式的唯一用途是将它们转换为功能接口。
这意味着如果方法接受函数接口 , 则可以传递lambda表达式 ,这进一步意味着,可以将lambda传递给所有接受方法的现有方法
Comparator
, Runnable
或任何其他只有一种抽象方法的接口。
这就是Java中lambda表达式也被称为SAM类型的原因,其中SAM表示单一抽象方法。
@Functional注释有什么作用?
现在,让我们看看@Functional
注释是做什么的? 如果仅在其上面添加@Functional
注释,它将使接口起作用吗? 好吧,不,它不会那样做。 实际上,
可选的 。
这意味着您可以在不使用@Functioanl
批注的情况下创建功能接口,就像可以在不将@Override
批注放在方法顶部的情况下@Override
方法一样。 那么, @Functional
注释的真正目的是什么?
好吧,它可以确保该接口实际上只有一种抽象方法,并且还向Javadoc之类的工具提供了此接口是功能性接口的提示。 它与@Override
注释非常相似,后者通过验证您实际上是重写方法来帮助防止人为错误。
与@Override
相似,它的最佳做法是使用单个抽象方法将@Functional
批注放在方法的顶部,以向Javadoc等工具表明它们是功能接口。
用@Functional
注释对java.util.function包中添加的所有新功能接口进行注释。
顺便说一句,是的,我们在JDK 8中提供了更多的功能接口,特别是通用功能接口,例如Predicate
, Supplier
, Consumer
, Function
, BiFunction
, UnaryOperator
等。请参见
Java 8:初学者的基础知识,可深入了解所有这些接口。
这些功能接口允许您以lambda表达式的形式将代码传递给函数,并允许创建可以对这些代码进行操作的强大方法,例如
filter()接受谓词,并允许您传递接受一个参数并返回布尔值的代码。
功能接口和Lamda表达式如何关联
功能接口的知识如何影响lambda表达式的编写? 好吧,除非您不了解功能接口,否则您将无法编写可转换为该功能接口的lambda表达式。
例如, java.util.Map
接口的merge()
方法接受BiFunction,但是如果您不知道什么是BiFunction,则无法为此编写lambda。
BiFunction
是一个功能接口,具有一个接受两个参数T和U并返回对象R的方法。
这意味着您可以将lambda传递给此方法,该方法可处理两个参数并返回一个对象,例如merge(key, value, (v1, v2) -> v1 + v2)
这里(v1, V2) -> v1 + v2
是一个可以转换为BiFunction功能接口实例的lambda表达式。
一个更简单的示例是Predicate
,它接受类型T并返回一个布尔值。 如果您看Stream类的filter()方法,它将接受一个谓词:
filter(Predicate predicate)
这意味着您可以传递任何接受一个参数的lambda表达式,然后将布尔值返回给该方法,例如age -> age > 15 or s -> s.length == 15
,这两个都是可接受的,但是如果您不知道什么是a谓词接口,那么您将无法做到这一点。
功能接口的另一个示例是Consumer
,它接受类型T的参数并且不返回任何内容。 JDK 8中Iterable的forEach()
方法很好地利用了这一点,如下所示:
forEach(Consumer action)
您会看到forEach()接受了Consumer,这意味着您可以向其传递一个lambda表达式,该表达式具有一个参数,不返回任何内容或返回空值,例如
s -> System.out.println(s)
代码System.out.println()
返回任何内容,它仅在控制台中打印行。
您可以看到,如果您知道功能接口,则可以轻松编写lambda表达式以进行传递,因此必须对功能接口有充分的了解。 我建议您遍历java.util.function
包中的所有功能接口并了解它们。
我将在以后的文章java.util.function
包中的一些更复杂的功能接口,但是如果您迫不及待,那么我建议您学习此Java 9 Master Class课程,以了解有关Lambda和其他Java的更多信息。 8个概念。
这就是Java中的功能接口 。 您还了解了@Functional
批注的作用以及为什么要对Java 8中的代码中的lambda表达式进行有效使用需要对功能接口有充分的了解。如果您尚未开始使用Java 8,建议您这样做。之所以这样做,是因为在未来几年中,每个人都将使用Java 8进行编码,如果您不知道lambda表达式和Java 8中引入的新功能,那么您将被抛在后面。
其他Java 8教程和参考资料,供进一步学习
Java 8的新增功能:Lambdas
如何使用Lambda按键和值对HashMay进行排序?
如何在Java 8中使用map和flatMap?
如何在Java 8中进行Map Reduce?
如何在Java 8中将lambda表达式转换为方法引用?
Java SE 8新功能–完整指南
感谢您到目前为止阅读本文。 如果您喜欢我对Functional接口和@Functional
注释的解释,请与您的朋友和同事分享。 如果您有任何疑问或反馈,请发表评论。
翻译自: https://www.javacodegeeks.com/2018/01/functional-interface-java-8-functional-annotation-examples.html