@FunctionalInterface
注解,那么编译器就会按照函数式接口的定义来要求该接口函数式接口应满足:
@FunctionalInterface
注解的接口应该满足函数式接口的定义如果不满足上述条件,编译器就会生成错误信息。
@FunctionalInterface
注解,那么编译器依旧会将该接口看作是函数式接口 如果一个接口声明了抽象方法,但该抽象方法重写了 Object 类里的一个公有方法,那么对于 Java 编译器来说,它并不会认为该方法符合函数式接口的抽象方法(即不把该方法当作函数式接口的抽象方法)。因为接口的实现类都会直接或间接继承 Object 这个根类,所以在函数式接口中定义与 Object 类中签名一样的方法是不会导致函数式接口失效的。
举例:
举例的接口声明为函数式接口,并重写了 toString()
这个 Object()
方法,但是并不会报错:
@FunctionalInterface
public interface MyInterface {
void test();
String toString();
}
jdk1.8 以后接口里面可以定义方法的实现,这种方法叫做 default-method。
举例:
举个例子:forEach方法
default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}
该实现方法定义在 Iterable 中。
jdk1.8 后接口也可以定义静态方法。
举例:
public static <T extends Comparable<? super T>> Comparator<T> naturalOrder() {
return (Comparator<T>) Comparators.NaturalOrderComparator.INSTANCE;
}
如果接口声明了 default 方法,并且某类实现了该接口,那么 default 方法将会被继承。
这里有个问题:
如果有一个类继承了两个不同接口的同名 default 方法,jvm 编译器是无法识别到底该使用哪个方法的,必须重写 default 方法,如下:
public class MyClass implements MyInterface1, MyInterface2 {
@Override
public void myMythod() {
MyInterface2.super.myMethod();
}
public static void main(String[] args) {
MyClass myClass = new MyClass();
myClass.myMethod();
}
}
可以重写也可以在重写方法中指定要继承接口的 default 方法。
实现类的优先级比接口高。
public class MyClass extends MyInterfaceImpl implements MyInterface2 {
public stativ void main(String[] args) {
MyClass myClass = new MyClass();
myClass.myMethod();
}
}
MyInterfaceImpl 中有个和 MyInterface2 同名的 myMethod()
方法。
运行上述代码,编译器并不会报错,因为实现类的优先级比接口高,所以 myClass.myMethod()
调用的方法默认是 MyInterfaceImpl 类中的实现方法。