Java不支持类的多重继承,但可通过接口多实现模拟;当多个接口含同签名default方法时,子类必须显式覆盖并可选择调用某接口实现;抽象类适合定义“是什么”及共享状态,接口适合定义“能做什么”。
Java中一个类只能用 extends 继承一个父类,这是硬性限制。试图写 class A extends B, C 会直接报错:「error: unexpected type」。真正能实现“多重行为复用”的路径是接口——接口可以多实现,且从 Java 8 起支持默认方法,从 Java 9 起支持私有方法,大幅提升了抽象能力。
当一个类同时 implements 多个接口,而这些接口都定义了同签名的 default 方法(如 void log()),编译器会拒绝编译,报错:「class A inherits unrelated defaults for log() from types X and Y」。必须由子类显式覆盖该方法。
X.super.log())就完事,仍需提供完整方法体X.super.log() 或 Y.super.log()
default 方法语义不同(比如一个打印日志,一个发告警),强制覆盖反而是设计提醒:你得明确当前类的行为意图interface Loggable {
default void log() { System.out.println("log"); }
}
interface Alertable {
default void log() { System.out.println("alert"); }
}
class Service implements Loggable, Alertable {
@Override
public void log() {
Loggable.super.log(); // 显式选择
}
}
接口适合定义“能做什么”,抽象类适合定义“是什么”以及共享状态和骨架逻辑。关键判断点在是否需要:
protected 成员变量或构造器(接口不允许)public,且 Java 9+ 才允许 private 辅助方法)init() 或构造块中)extends 占位,此时只能靠接口补充能力接口里的 static 方法属于接口自身,不是实现类的成员。子类无法通过 this.method() 或重写它,但可通过 InterfaceName.method() 直接调用。这点常被误认为“继承”,其实只是命

static 方法里Utils.xxx()),而非塞进某个接口static 方法不能访问 this,也不能被 default 方法隐式代理(除非显式传参)interface Codec {
static String encode(String s) { return Base64.getEncoder().encodeToString(s.getBytes()); }
}
// 正确调用:
String x = Codec.encode("hello");
// 错误写法(编译不过):
// new MyClass().encode("hello");
接口的 default 方法和实现类的继承链之间没有动态绑定关系,它们只是编译期静态解析的契约补丁。真要模拟“多重继承语义”,得靠组合+委托,而不是依赖 interface 的语法糖。