前言
关于函数式接口都可以使用lambda表达式实现
具体涉及到lambda表达式可以看我之前的文章进行预习
java中lambda表达式
言归正传@FunctionalInterface
该注解主要是:一种信息性注释类型,用于指示接口类型声明是Java语言规范中定义的功能接口。
1. 函数式接口
函数式接口是只包含一个抽象方法声明的接口
java.lang.Runnable 就是一种函数式接口,在 Runnable 接口中只声明了一个方法 void run()
具体关于这个类可看我对线程的见解
java线程的三种创建方式详细分析(全)
常用的一些接口Callable、Runnable等在JDK8中都添加了@FunctionalInterface注解
而且这些线程的接口在源码上有该注解
如图所示
那我们就来分析一下它是什么含义
2. @FunctionalInterface定义
通过该注解的源代码可看到其代码注释
翻译过来其要点有如下:
- 一个函数式接口有且只有一个精确的抽象方法。可以有多个静态方法或者默认方法, 每一个该类型的lambda表达式都会被匹配到这个抽象方法
- 从概念上讲,一个功能接口只有一个抽象方法。 因为默认方法有实现,所以它们不是抽象的。 如果一个接口声明了一个抽象方法,覆盖了java.lang的一个公共方法。 对象,它也不计入接口的抽象方法计数,因为接口的任何实现都会有来自java.lang.Object或其他地方的实现
- 函数接口的实例可以通过lambda表达式、方法引用或构造函数引用创建
如果接口上只有一个抽象方法,但我们没有对其加上@FunctionalInterface 编译器仍然将其看作函数式接口。该注解是给编译器做检查使用的,如果使用了该注解,编译器就会检查该接口中的抽象方法是不是只有一个,如果有多个就会报错
通过@FunctionalInterface
其源码可看到
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface FunctionalInterface {}
具体其源码中的注解为
- 使用
@Documented
标注了,在生成javadoc的时候就会把@Documented注解给显示出来,但其实也没啥用处,一个标识而已 @Retention
作用是定义被它所注解的注解保留多久,RetentionPolicy.RUNTIME
注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在@Target
作用于接口中
对于Target的补充:
参数 | 描述 |
---|---|
CONSTRUCTOR | 构造器 |
FIELD | 域 |
LOCAL_VARIABLE | 局部变量 |
METHOD | 方法 |
PACKAGE | 包 |
PARAMETER | 参数 |
TYPE | 类、接口(包括注解类型) 或enum声明 |
3. 使用方法
具体如何使用该代码,只需要在接口上定义该注解即可
@FunctionalInterface
public interface Interface {
public void do();
}
如果使用两个接口就会出错
//错误展示
@FunctionalInterface
public interface Interface {
public void do();
public void love();
}
正确展示
//正确展示
@FunctionalInterface
public interface Interface {
public void do();
// java.lang.Object中的方法不是抽象方法
public boolean equals(Object var);
// default不是抽象方法
public default void defaultlove(){
}
// static不是抽象方法
public static void love(){
}
}
通过如上方法
可以确信这个注解的正确规则为
- 注解在接口上
- 被注解的接口有且只有一个抽象方法
- 被注解的接口可以有默认方法/静态方法,或者重写Object的方法