在jdk8之前,interface之中可以定义变量和方法,变量必须是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及以后,允许我们在接口中定义static方法和default方法。

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");  
    }  
}

注意:

  • 接口中定义的静态static方法只能通过接口名直接调用,default的方法需要用接口的实现类的对象来调用
  • 接口中的static和default方法可以有函数体,其实现类不必要重写
  • 其他的非static和非default的都是抽象方法,没有函数体,其实现类必须重写所有的抽象方法
  • 如果子类(或实现类)继承的父类和其实现的接口定义了同名同参的方法,并且接口中的方法为default方法(都有函数体),那么该子类的对象调用该方法时(在子类没有重写该方法的情况下),默认是父类的方法(类优先性)
  • 如果类实现了多个接口,而且多个接口中定义了同名同参数的default方法(有函数体),在该类没有重写的情况下,就会报错(接口冲突)。如果想解决这个问题,就必须在该类中重写此方法。

由于java支持一个实现类可以实现多个接口,如果多个接口中存在同样的static和default方法会怎么样呢?如果有两个接口中的静态方法一模一样,并且一个实现类同时实现了这两个接口,此时并不会产生错误,因为jdk8只能通过接口类调用接口中的静态方法,所以对编译器来说是可以区分的。但是如果两个接口中定义了一模一样的默认方法,并且一个实现类同时实现了这两个接口,那么必须在实现类中重写默认方法,否则编译失败。