JDK8中接口的变化
那么多学技术的都可以成功,凭什么我不行
在JDK8之前,接口之中可以定义变量和方法。
变量必须(默认)是public、static、final的,方法必须(默认)是public、abstract的。
由于这些修饰符都是默认的以下写法等价:
public interface JDK8BeforeInterface {
public static final int field1 = 0;
int field2 = 0;
public abstract void method1(int a) throws Exception;
void method2(int a) throws Exception;
}
在JDK8之后,接口中可以有默认的方法实现,和静态方法。
public interface JDK8Interface {
// static修饰符定义静态方法
static void staticMethod() {
System.out.println("接口中的静态方法");
}
// default修饰符定义默认方法
default void defaultMethod() {
System.out.println("接口中的默认方法");
}
}
public class JDK8InterfaceImpl implements JDK8Interface {
//实现接口后,因为默认方法不是抽象方法,所以可以不重写,但是如果开发需要,也可以重写
}
public class Main {
public static void main(String[] args) {
// static方法必须通过接口类调用
JDK8Interface.staticMethod();
//default方法必须通过实现类的对象调用
new JDK8InterfaceImpl().defaultMethod();
}
}
当然如果接口中的默认方法不能满足某个实现类需要,那么实现类可以覆盖默认方法。
public class AnotherJDK8InterfaceImpl implements JDK8Interface {
// 签名跟接口default方法一致,但是不能再加default修饰符
@Override
public void defaultMethod() {
System.out.println("接口实现类覆盖了接口中的default");
}
}
由于java支持一个实现类可以实现多个接口,如果多个接口中存在同样的static和default方法会怎么样呢?
如果有两个接口中的静态方法一模一样,并且一个实现类同时实现了这两个接口,此时并不会产生错误,因为JDK8只能通过接口类调用接口中的静态方法,所以对编译器来说是可以区分的。
但是如果两个接口中定义了一模一样的默认方法,并且一个实现类同时实现了这两个接口,那么会产生默认方法冲突,编译器会提示接口冲突,让我们自行解决,那么必须在实现类中重写默认方法,否则编译失败。
当一个接口实现另一个接口时,子接口与父接口存在相同的默认方法时,将采用父类接口的默认方法。
当一个类既继承父类又实现接口时,如果父类中的方法与接口中的默认方法冲突,将会执行父类中的方法,而不是接口中的方法,这种情况被称为类优先选择,方便兼容JDK8之前的版本。