接口默认方法
在 Java 8 之前,接口允许定义public static final
常量和public abstract
方法,即使不显式写public static final
或public abstract
,编译器也会自动添加。同时,不允许定义方法实现(即非抽象方法)。
由于接口中的方法只能是public static final
,所以所有实现这个接口的类都必须实现这些方法。这就导致一个严重的问题:
一旦接口发布,就不能轻易。否则所有实现类都必须修改、编译,否则就会编译失败。
这对于大型框架或公共 API 是灾难性的:每次新增方法都可能破坏成百上千个实现类。
default 方法
Java 8 引入 default
方法,允许在接口中提供方法的默认实现。
这种方法可以允许方法有自己的默认实现,而不需要子类去覆盖它。已有的实现类不需要修改就能继续工作,如果它们想要自定义行为,也可以选择性地重写默认方法。
interface Animal {
void speak();
default void walk() {
System.out.println("Animal walking...");
}
}
解决冲突
-
如果一个类实现了多个接口,且这些接口都有相同签名的默认方法,必须在实现类中解决冲突(明确重写)。
interface A { default void hello() { System.out.println("Hello from A"); } } interface B { default void hello() { System.out.println("Hello from B"); } }
❌ 直接实现两个类接口会报错:
class C implements A, B { // 编译错误:类 C 继承了 A 和 B 中的 hello(),冲突 }
✅ 正确做法:必须显式地重写方法:
class C implements A, B { @Override public void hello() { // 可以选择调用其中一个接口的默认实现 A.super.hello(); // 或 B.super.hello(); } }
-
如果一个类继承的父类和实现的接口有相同的方法,“类优先与接口”原则,但可以重写。
class Parent { public void hello() { System.out.println("Hello from Parent"); } } interface A { default void hello() { System.out.println("Hello from A"); } } class C extends Parent implements A { // 不需要重写,使用 Parent 的 hello() }
📌 多接口默认方法冲突的解决规则总结:
情况 | 是否必须重写 | 原因 |
---|---|---|
只有一个接口提供 default 方法 | ❌ 不需要 | 没有冲突 |
多个接口提供同名 default 方法 | ✅ 必须 | 编译错误,必须显式重写并选择调用哪个接口的方法 |
一个接口提供 default ,另一个是类方法 | ✅ 类的方法优先 | “类优先于接口” 原则,但可以重写 |
接口静态方法
接口中的static
方法是带有实现的方法,但是只能通过接口名调用,不能被实现类继承或重写。
通过 static 方法可以在接口中封装和该接口相关的工具函数,保留与接口相关的静态逻辑在接口中,而不是放到外部类或工具类中,提高封装性。同时,不像抽象类的静态方法可能被子类继承,接口的静态方法只属于自己,避免污染实现类。
示例:
interface MyInterface {
static void staticMethod() {
System.out.println("Static method in interface");
}
}
调用方式:
public class Test {
public static void main(String[] args) {
MyInterface.staticMethod(); // ✅ 正确调用
}
}
🚫 不能通过实现类或实现对象调用:
class MyClass implements MyInterface {}
MyClass obj = new MyClass();
obj.staticMethod(); // ❌ 编译错误,不能通过对象调用接口的 static 方法
在 Java 8 中,接口中的 static 方法必须是
public
,默认就是public
,写或不写都一样。不能是private
或protected
。如果你想用private static
方法,只能在 Java 9 或更高版本中使用。