在Java 8中,接口(Interface)得到了两个重大的增强:默认方法(Default Methods)和静态方法(Static Methods)。这些特性为接口的实现提供了更大的灵活性和功能。
默认方法
- 默认方法允许接口中有具体实现的方法,这意味着你可以在不破坏实现类的情况下,为接口添加新的方法。这对于维护已有代码非常有用,因为你可以在不修改现有实现类的情况下,为接口添加新的功能。
- 默认方法用
default
关键字修饰,子类的实例可以调用default
方法。
静态方法
- 静态方法使得接口中可以有静态的实现,这对于工具类接口非常有用。静态方法可以直接通过接口名调用,而不需要接口的实例。
- 静态方法通过接口名调用。
BaseInterface.java
public interface BaseInterface {
void normalMethod();
default void defaultMethod() {
System.out.println("我是默认方法,可以写方法体");
}
static void staticMethod() {
System.out.println("我是静态的方法,可以写方法体");
}
}
BaseInterfaceImpl.java
package com.example.jdk8;
public class BaseInterfaceImpl implements BaseInterface {
@Override
public void normalMethod() {
System.out.println("我是普通方法");
}
}
Main.java
public class Main {
public static void main(String[] args) {
BaseInterface baseInterface = new BaseInterfaceImpl();
baseInterface.normalMethod();
baseInterface.defaultMethod();
BaseInterface.staticMethod();
}
}
运行结果:
我是普通方法
我是默认方法,可以写方法体
我是静态的方法,可以写方法体
多个接口中存在同样的static和default方法
由于JAVA支持一个实现类可以实现多个接口,如果多个接口中存在同样的static和default方法会怎么样呢?
- 如果有两个接口中的static方法一模一样,并且实现类同时实现了这两个接口,此时并不会产生错误,因为静态方法只能通过接口类调用,所以对编译器来说是可以区分的。
- 如果两个接口中定义了一模一样的default方法,并且一个实现类同时实现了这两个接口,那么必须在实现类中重写默认方法,否则编译失败。
public interface A {
/** default访问修饰符修饰的方法 */
default void methodB1() {
System.out.println("this is default method -- InterfaceA");
}
/** static修饰符修饰的方法 */
public static void methodB2() {
System.out.println("this is static method -- InterfaceA");
}
}
public interface B{
/** default访问修饰符修饰的方法 */
default void methodB1() {
System.out.println("this is default method -- InterfaceB");
}
/** static修饰符修饰的方法 */
public static void methodB2() {
System.out.println("this is static method -- InterfaceB");
}
}
public class Test implements A,B{
/** 由于A和B中default方法一样,所以这里必须覆盖 */
@Override
public void methodB1() {
System.out.println("this is Overriding methods");
}
public static void main(String[] args) {
/** 接口的default方法,通过接口的实现类的对象调用 */
Test test = new Test();
test.methodB1();
/** A接口的静态方法,通过接口名调用 */
A.methodB2();
/** B接口的静态方法,通过接口名调用 */
B.methodB2();
}
}