继承表达“is-a”关系,如Dog是Animal;误用会导致语义错误和扩展困难;子类构造时父类先初始化;private成员不参与继承;Java只支持单继承,应结合组合与接口使用。

父类和子类是“is-a”关系,不是“has-a”或“uses-a”
Java中继承表达的是严格的“是一个”语义:比如 Dog 是一个 Animal,Car 是一个 Vehicle。这种关系决定了设计起点——不能因为两个类“都有 name 字段”就强行让它们继承同一个父类;必须真实存在分类学或业务逻辑上的上下位关系。
- 反例:
Student和Course都有id,但学生不是课程,不能继承 - 正例:
Penguin和Mouse都属于动物,共用name、eat()等,适合提取为Animal父类 - 一旦误用继承(比如为复用字段而硬套),后续扩展会出问题:子类被迫继承无意义的方法,或重写时破坏语义(如让
Course.eat()抛异常)
子类对象创建时,父类部分先初始化
执行 new Dog("Buddy") 时,JVM 不是直接构造 Dog,而是先调用父类 Animal 的构造方法(隐式或显式),再执行子类自己的初始化逻辑。这决定了字段赋值顺序、this 可见性,以及为什么子类构造器第一行必须是 super(...) 或 this(...)。
- 若父类只有带参构造器,子类必须显式调用
super(name),否则编译失败 - 子类中访问同名字段时,
name默认指子类字段,super.name才能访问父类字段(即使父类字段是protected) - 在父类构造器中调用被子类重写的方法,会导致
NullPointerException或未预期行为(因为子类字段尚未初始化)
继承只能访问非私有成员,private 字段/方法不参与继承
private 成员对子类完全不可见,既不能直接访问,也不能被重写。很多初学者误以为“继承了所有字段”,结果在子类里写 this.age 报错,其实是因为父类的 age 是 private。
- 正确做法:父类提供
public或protected的 getter/setter,或把字段声明为protected - 子类无法重写
private方法,哪怕签名相同,也只是定义了一个新方法,跟父类无关 - 想让子类安全扩展行为?用
protected+ 模板方法模式,而不是暴露private细节
Java只支持单继承,但可通过组合+接口弥补局限
一个类只能 extends 一个父类,这是语言强制约束。它避免了多继承的菱形问题,但也意味着你不能同时“是一个 Animal”又“是一个 Machine”。这时候别硬改继承链,该用组合就用组合。
立即学习“Java免费学习笔记(深入)”;
- 例如:机器人狗既要会叫(Animal 行为),又要充电(Machine 行为),就让
RobotDog持有AnimalBehavior和PowerSource实例,而不是试图多继承 - 接口(
interface)可以多实现,适合定义能力契约(如Flyable、Swimmable),和继承正交使用 - 过度依赖继承容易导致类层次过深,
Animal → Mammal → Canine → Dog → Labrador这种五层继承,在实际项目中往往比扁平化+组合更难维护
真正难的不是写 extends 关键字,而是判断某个共性该抽成父类、接口,还是干脆不抽——这取决于它是否代表一种稳定的分类关系,而不是临时的代码相似。










