接口是公开、抽象、可组合的行为契约,表达“能力”而非“身份”,支持多实现与向后兼容演进;abstract class 表达纵向类型继承,仅单继承。

Java 中的 interface 本身不能被实例化,也不包含状态(字段只能是 public static final),更不提供默认实现(除非用 default 或 static 方法)。它的本质是一组**公开、抽象、可组合的行为约定**——编译器强制实现类必须提供这些方法的具体逻辑,但不管你怎么实现。
这种设计让接口天然适合表达「能力」而非「身份」。比如一个类可以同时 implement Comparable 和 Serializable,这不是在说“它是什么”,而是在声明“它能做什么”和“它支持什么协议”。
二者都支持抽象方法,但语义和用途完全不同:
interface 表达的是横向能力扩展,支持多继承(一个类可实现多个接口);abstract class 是纵向类型继承,只允许单继承public abstract,字段默认是 public static final;抽象类可含 protected 方法、构造器、非 final 字段default 方法(提供默认行为,不破坏已有实现),但无法拥有实例状态;抽象类的 default 行为天然可访问 this 和成员变量Proxy.newProxyInstance() 只接受接口列表,不接受抽象类给已有接口添加新方法会导致所有实现类编译失败。Java 8 引入 default 方法,就是为了解决接口的**向后兼容演进问题**。但它不是让你把接口当抽象类用:
default 方法应仅提供通用、无状态、不依赖实现细节的逻辑(如集合操作的 stream()、forEach())default 方法里调用其他 default 方法形成隐式依赖链super.xxx() 堆叠行为default 方法无法访问 private 成员(接口里不允许定义 private 实例字段)public interface EventProcessor {
void handle(Event e);
default void batchHandle(List events) {
events.forEach(this::handle); // 安全:只调用自身已承诺的抽象方法
}
}
很多团队把接口写成“方法大杂烩”,结果导致实现类负担过重或语义模糊。真正关键的约束不在语法,而在设计意图:
Runnable 只管执行,Callable 只管执行并返回,不混在一起Collection 而非 ArrayList),避免因具体实现变更被迫改接口Consumer 和 Consumer 在运行时是同一个类型,无法做 instanceof 判断接口一旦发布,修改成本远高于类——哪怕加一个 default 方法,也要考虑老实现是否意外覆盖了同名方法,或子类是否因多继承产生 default 冲突。