继承表达类型归属关系而非属性复制,子类对象可赋值给父类引用但反之不行,方法调用依运行时类型;构造器不继承但须显式或隐式调用super();protected跨包仅限子类自身访问;@Override是编译期契约而非装饰。

继承的本质是“是什么”而不是“有什么”
初学者常把 extends 理解成“复制父类字段和方法”,结果在子类里盲目重写 toString() 或乱调 super(),却不知道为什么编译报错。其实继承表达的是类型归属关系:比如 Dog 是一种 Animal,不是“Dog 拥有 Animal 的属性”。这意味着:
– 子类对象可以直接赋值给父类引用(Animal a = new Dog(););
– 但反过来不行(Dog d = new Animal(); 编译失败);
– 方法调用走的是运行时实际类型(多态),不是声明类型。
构造器不会被继承,但必须显式或隐式调用
很多人写子类构造器时忘记处理父类初始化,导致编译错误 Implicit super constructor Animal() is undefined。Java 规定:每个子类构造器第一行,要么显式写 super(...),要么隐式插入 super()——但如果父类没有无参构造器,隐式调用就会失败。
class Animal {
Animal(String name) { this.name = name; }
}
class Dog extends Animal {
Dog(String name) {
super(name); // 必须写!否则编译不过
}
}
常见错误:
– 父类只有带参构造器,子类构造器里没写 super(...);
– super(...) 不在第一行(哪怕前面加个 System.out.println 都会报错);
– 在子类构造器中调用 this(...) 后又试图写 super(...)(二者互斥)。
protected 和 package-private 的可见性容易误判
初学者看到 protected 就以为“子类 anywhere 都能访问”,结果在不同包里的子类中访问父类 protected 字段失败。真实规则是:
– protected 成员可在同一包内任意访问;
– 跨包时,**仅允许通过子类自身对象访问**(不能通过父类引用或其它子类实例);
– default(包私有)跨包完全不可见,哪怕有继承关系也没用。
立即学习“Java免费学习笔记(深入)”;
例如:
– new Dog().age(age 是 protected int age)✓ 可以;
– Animal a = new Dog(); a.age ✗ 编译错误;
– new Cat().age(Cat 也是 Animal 子类,但和 Dog 不同文件)✗ 不可访问。
@Override 不是装饰,是编译期契约
加上 @Override 注解不是为了“好看”或“让 IDE 提示”,而是告诉编译器:“我声明这个方法一定存在于父类/接口中”。一旦父类删掉该方法,或签名稍有不同(比如参数类型从 List 改成 ArrayList),编译就直接报错,避免静默失效。
典型陷阱:
– 重写 equals(Object) 时写成 equals(String),漏掉 @Override 就变成重载,毫无多态效果;
– 在实现接口时没加 @Override,后来接口新增默认方法,自己写的同名方法被当成新方法而非实现,逻辑跑飞;
– 用 Lombok 的 @Data 自动生成 toString(),再手动写一个不加 @Override 的同名方法,结果两个都存在,运行时调哪个全看顺序。
move() 可以通用,但 bark() 就不该在 Animal 里定义——强制所有动物都会叫,企鹅和树懒就只能抛异常。这种建模分寸,比记住 super 怎么写更需要反复试错。










