抽象方法必须定义在abstract类中且无方法体,以分号结尾;不能用private、static、final修饰;接口中java 8+的抽象方法默认public且不可显式加abstract关键字。

抽象方法必须定义在 abstract 类里,且不能有方法体
抽象方法只声明不实现,语法上以分号结尾,abstract 修饰符不可省略。它强制子类重写,但编译器不会帮你检查是否真的被覆盖——如果子类没实现且自己也不是 abstract,编译直接报错:error: class X must either be declared abstract or implement abstract method Y in Z。
常见错误是误把抽象方法写成空方法体(比如加了 {}),这会让编译器当成普通方法,失去抽象语义;或者忘了在类名前加 abstract,导致“非法的抽象方法声明”错误。
- 抽象方法只能出现在
abstract class中,不能在普通类或接口中(Java 8+ 接口里叫默认方法或静态方法,不是抽象方法) - 抽象方法不能用
private、static、final修饰——它们和“强制子类实现”冲突 - 抽象方法可以有访问修饰符(
public、protected),但不能是包私有(即不写修饰符)+abstract组合,否则编译报错
接口方法默认就是 public abstract,但 Java 8 后行为已分化
Java 8 之前,接口里的所有方法都隐式是 public abstract,连修饰符都不能写;Java 8 起,接口可定义 default 方法(带实现)和 static 方法(属于接口本身),只有未加修饰符的方法才仍是抽象的——但此时你依然不能写 abstract 关键字,写了反而编译失败:modifier abstract not allowed here。
关键区别在于:接口抽象方法不要求实现类“继承”,而是“实现”;而抽象类的抽象方法靠“继承+重写”来兑现。这意味着一个类可以实现多个接口,但只能继承一个抽象类。
立即学习“Java免费学习笔记(深入)”;
- 接口中的抽象方法始终是
public,哪怕不写——也不能用protected或包私有 -
default方法可被子接口重写,也可被实现类重写;但若实现类没重写,又没提供自己的实现,就沿用接口默认逻辑 - 抽象类可以有构造器、成员变量、非抽象方法;接口不行(Java 9+ 允许私有方法辅助
default,但仍是工具性质)
普通方法有完整实现,但受继承规则约束
普通方法(即非抽象、非静态、非 final 的实例方法)的核心是“可被继承并可能被重写”。它的行为取决于运行时实际类型(动态绑定),但能否被访问、能否被重写,由修饰符决定:
-
private方法不能被继承,子类里同名方法只是新定义,跟父类无关 -
final方法可被继承,但禁止重写;强行重写会触发编译错误:error: cannot override final method -
static方法属于类,不是实例,子类里同名方法是隐藏(hiding),不是重写(overriding);调用取决于引用类型,而非实际对象类型 - 普通方法若在抽象类中,不影响类的抽象性;它只是“白送”的能力,子类可直接用,也可选择重写
选抽象类还是接口?看你要表达的是“是什么”还是“能做什么”
抽象类强调共性建模:比如 Animal 是个抽象类,Dog 和 Cat 是它的子类,它们共享生命状态、呼吸方式等基础逻辑;接口强调能力契约:比如 Flyable、Swimmable,一个 Duck 可以同时实现二者,但 Ostrich 只实现后者——这种组合能力抽象类做不到。
真正容易被忽略的是:从 Java 8 开始,接口也能提供行为(default 方法),但它仍不能保存状态(没有实例字段)。一旦你需要初始化字段、控制构造流程、或复用部分逻辑(比如模板方法模式),抽象类就不可替代。
另一个现实坑点:如果已有类想新增某个能力,但该类已继承了别的类,那只能走接口路线——因为 Java 不支持多继承,但允许实现多个接口。











