静态方法和默认方法都必须有自己的实现,如果一个类既有静态方法,又有默认方法,还有一个抽象方法,那么它还是一个函数式接口。
Java8为接口引入了静态方法功能。静态方法必须有自己的实现,而不能仅仅是一个接口声明,不然编译是会报错的。
以前为一个接口编写一个工具类,需另外写一个工具类如XxxUtil,现在可以直接将工具类的方法直接写在接口的静态方法中了。后面我们会看到JDK为我们提供的工具方法大部分都在函数式接口中,如java.util.function.Function。
package java.util.function;
import java.util.Objects;
@FunctionalInterface
public interface Function<T, R> {
R apply(T t);
default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
Objects.requireNonNull(before);
return (V v) -> apply(before.apply(v));
}
default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
Objects.requireNonNull(after);
return (T t) -> after.apply(apply(t));
}
static <T> Function<T, T> identity() {
return t -> t;
}
}
函数式接口Function提供一个抽象方法(apply),两个默认方法(compose和andThen),一个静态方法(identity)。
接口中的方法默认都是抽象的和public的。
问题:java 8之前接口增加一个新的方法,所有的实现类都得实现此方法。
为了解决这个问题,java8为接口引入了默认方法,如果接口新增了一个默认方法,所有的实现类无需实现此方法(像继承类的方法一样继承了默认方法的具体实现)。
默认方法由default修饰符修饰,并像类中声明的其他方法一样包含方法体。
package com.morris.java8.defaultmethod;
public class DefaultMethodExample {
public static void main(String[] args) {
A a = () -> 10;
System.out.println(a.isEmpty());
}
@FunctionalInterface
interface A {
int size();
default boolean isEmpty() {
return 0 == size();
}
}
}
如果一个类使用相同的函数签名从多个地方(比如另一个类或接口)继承了方法,通过三条规则可以进行判断。
类中的方法优先级最高。类或父类中声明的方法的优先级高于任何声明为默认方法的优先级。
如果无法依据第一条进行判断,那么子接口的优先级更高:函数签名相同时,优先选择拥有最具体实现的默认方法的接口,即如果B继承了A,那么B就比A更加具体。
最后,如果还是无法判断,继承了多个接口的类必须通过显式覆盖和调用期望的方法,显式地选择使用哪一个默认方法的实现。
类的方法优先级最高:
package com.morris.java8.defaultmethod;
public class MultiExtendExample {
public static void main(String[] args) {
new C().hello(); // Hello from C
}
public interface A {
default void hello() {
System.out.println("Hello from A");
}
}
public interface B extends A {
@Override
default void hello() {
System.out.println("Hello from B");
}
}
public static class C implements B, A {
@Override
public void hello() {
System.out.println("Hello from C");
}
}
}
最具体的实现优先:
package com.morris.java8.defaultmethod;
public class MultiExtendExample1 {
public static void main(String[] args) {
new C().hello(); // Hello from B
}
public interface A {
default void hello() {
System.out.println("Hello from A");
}
}
public interface B extends A {
@Override
default void hello() {
System.out.println("Hello from B");
}
}
public static class C implements B, A {
}
}
冲突依旧无法解决,只能在类中覆盖该默认方法:
package com.morris.java8.defaultmethod;
public class MultiExtendExample2 {
public static void main(String[] args) {
new D().hello(); // Hello from D
}
public interface A {
void hello();
}
public interface B extends A {
@Override
default void hello() {
System.out.println("Hello from B");
}
}
public interface C extends A {
@Override
default void hello() {
System.out.println("Hello from C");
}
}
public static class D implements B, C {
@Override
public void hello() {
System.out.println("Hello from D");
}
}
}