抽象类和抽象方法必须用abstract修饰;含抽象方法的类须声明为abstract,abstract类可无抽象方法;abstract方法不能是private、static或final;抽象类不可实例化,但可用多态引用子类对象。

抽象类和抽象方法必须用 abstract 修饰,否则编译不通过
Java 中,只要类里有一个抽象方法,这个类就必须声明为 abstract;反过来,abstract 类可以没有抽象方法,但一旦有,子类必须实现(除非子类也声明为 abstract)。常见错误是忘记加 abstract 关键字,比如写成 public void makeSound(); 而不是 public abstract void makeSound();,这会导致编译报错:error: abstract method cannot have a body 或 error: missing method body, or declare abstract。
正确写法示例(动物叫):
abstract class Animal {
public abstract void makeSound(); // ✅ 必须 abstract
}
class Dog extends Animal {
@Override
public void makeSound() {
System.out.println("Woof!");
}
}
abstract 方法不能是 private、static 或 final
抽象方法本质是“强制子类提供实现”,所以它必须能被子类继承并重写。而 private 方法不可见,static 方法属于类而非实例,final 方法禁止重写——三者都与抽象方法的设计目的冲突。如果误写成 private abstract void draw();,JDK 直接拒绝编译,报错:illegal combination of modifiers: abstract and private。
图形计算案例中,常见错误写法:
立即学习“Java免费学习笔记(深入)”;
// ❌ 错误:abstract + static 不允许 public static abstract double getArea(); // ✅ 正确:只保留 abstract,由子类实例调用 public abstract double getArea();
-
getArea()应在子类如Circle、Rectangle中用实例字段(如radius、width)计算 - 如果真需要静态计算逻辑,应放在具体子类里作为辅助方法,或用工厂类封装
- 抽象方法默认是
public,不写访问修饰符会编译失败(不允许default或包私有)
抽象类不能直接 new,但可以用多态方式持有子类实例
试图执行 new Animal() 会触发编译错误:Animal is abstract; cannot be instantiated。但你可以用抽象类类型声明引用,指向具体子类对象,这是多态的核心用法。例如统一处理不同动物叫声或图形面积时,这种写法既安全又灵活。
实操建议:
Animal a1 = new Dog(); Animal a2 = new Cat(); a1.makeSound(); // Woof! a2.makeSound(); // Meow! Shape s1 = new Circle(5.0); Shape s2 = new Rectangle(4.0, 6.0); System.out.println(s1.getArea()); // 78.539... System.out.println(s2.getArea()); // 24.0
- 所有子类必须实现全部抽象方法,漏掉一个(哪怕只是忘了
@Override)就会导致子类编译失败 - 构造器在抽象类中允许存在,用于初始化共用字段,但只能被子类
super()调用 - 抽象类可以包含普通方法、静态方法、字段,这些可被子类直接继承使用
接口 vs 抽象类:别为了“抽象”而强行用 abstract class
如果只是定义行为契约(比如“能叫”“能画”),且不需要共享状态或非抽象方法实现,优先用 interface。从 Java 8 开始,接口也能有 default 和 static 方法,覆盖很多过去依赖抽象类的场景。
比如“图形绘制”需求,若只需统一 getArea() 和 getPerimeter(),用接口更轻量:
interface Shape {
double getArea();
double getPerimeter();
}
class Circle implements Shape { ... }
- 类只能单继承抽象类,但可实现多个接口——设计上更松耦合
- 抽象类适合表达“is-a”关系并复用代码(如所有
Animal共享name字段和sleep()方法) - 实际项目中,抽象类常配合模板方法模式(
templateMethod()调用若干abstract步骤),这点接口做不到
最容易被忽略的是:抽象类的字段可以是 protected 或包级,供子类直接访问;而接口中所有字段隐式为 public static final,只能是常量。










