Java开发人员应该对java.lang.Runnable,java.util.Comparator,java.util.concurrent.Callable
等等接口不会感到陌生。他们都只有一个单一的抽象方法。这样的接口,我们通常叫单一抽象方法接口(SAM,Single Abstract Method Interface)。
以前大家应该经常使用下面的代码片段
public class InnerAnonymousClassSample { public static void main(String[] args){ new Thread(new Runnable() { @Override public void run() { System.out.println("Anonymous Class Thread Demo"); } }).start(); }}
在Java 8
中,对于这种拥有单一抽象方法的接口,换了个名字,叫函数接口。所以,这个不是什么新东西,这个名字也是为了Lambada表达式而生。还是拿Runnable来说事,看看和Lambda表达式合壁后的结果。
public class ThreadWithLambda { public static void main(String[] args){ new Thread(() -> System.out.println("Thread with Lambda expression Demo")).start(); } }
回到函数接口上来,Java引入了@FunctionalInterface注解。用了它,你定义的接口就得是只有一个抽象方法的函数接口,否则会报编译错误。
@FunctionalInterface public interface FunInterfaceAnnotationDemo { public void absMethodDemo(); public void absMoreMethodDemo(); }
Error:(8, 1) java: 意外的 @FunctionalInterface 注释 com.tr.learning.lambda.FunInterface.FunInterfaceAnnotationDemo 不是函数接口 在 接口 com.tr.learning.lambda.FunInterface.FunInterfaceAnnotationDemo 中找到多个非覆盖抽象方法
上述方法可以通过Java 8中引入的关键字default 在接口中为其中一个抽象方法指定一个默认实现,这样上述接口就会变成合法的函数接口了。
@FunctionalInterface public interface FunInterfaceAnnotationDemo { public void absMethodDemo(); default public void absMoreMethodDemo(){ System.out.println("Default implementation of method ininterface"); }
一旦有了default
关键字,现在的接口就好像变成了抽象类。这让人总觉得怪怪的。这个default的接口主要是为了向后兼容。当你往一个拥有众多子类的接口中添加方法时,你总不想一个子类一个子类的去把方法实现吧?这在Java
JDK不断升级的过程中,是一个很现实的问题。至于说使用带default实现的接口还是抽象类,我认为在进行Java代码的设计时,还是抽象类更合适。
时间: 2024-10-10 12:22:29