必须为重写方法添加@Override注解以避免静默失效;向上转型后只能调用父类声明的方法;构造器中调用可重写方法会导致子类字段未初始化;static方法不参与多态,仅按声明类型绑定。

子类重写方法但没加 @Override 注解
编译器不会报错,但容易因方法签名微小差异(比如参数类型写成 int 而非 Integer)导致实际未重写父类方法,调用时仍走父类逻辑。这不是运行时错误,而是静默失效。
- 务必对所有明确意图重写的方法添加
@Override,让编译器校验签名一致性 - IDE 通常能自动提示缺失注解;若手动补上后报错,说明方法名、参数列表或返回类型不匹配
- 注意:Java 8 开始接口默认方法也可被
@Override,但static方法不能重写,加注解会直接编译失败
向上转型后调用子类特有方法失败
把子类实例赋给父类引用(如 Animal a = new Dog();),此时只能访问 Animal 中声明的方法。试图调用 Dog 独有的 bark() 会编译报错:cannot resolve method 'bark()'。
- 这是编译期检查,不是多态本身的问题,而是类型安全机制在起作用
- 若确实需要调用子类方法,必须向下转型:
((Dog) a).bark(); - 但强制转型前应先用
instanceof判断,否则可能抛ClassCastException
构造器中调用可被重写的方法
在父类构造器里调用一个被子类重写的方法,会导致子类中该方法在子类字段还未初始化时就被执行——字段值为默认值(如 null、0),极易引发空指针或逻辑异常。
class Parent {
String name = "parent";
Parent() {
printName(); // 实际调用的是子类的 printName()
}
void printName() {
System.out.println(name);
}
}
class Child extends Parent {
String name = "child";
@Override
void printName() {
System.out.println(name); // 输出 null!因为此时 Child.name 还没初始化
}
}
- 避免在构造器中调用
public或protected的非final方法 - 更安全的做法是将该方法设为
private或final,确保无法被重写 - 若必须延迟初始化,考虑用工厂方法或构建器模式替代直接在构造器中触发行为
静态方法和属性不参与多态
看似“重写”了父类的 static 方法,但实际只是隐藏(hiding)。通过父类引用调用时,绑定的是声明类型,而非运行时类型——这和实例方法的动态绑定完全不同。
立即学习“Java免费学习笔记(深入)”;
class A {
static void show() { System.out.println("A"); }
}
class B extends A {
static void show() { System.out.println("B"); }
}
A a = new B();
a.show(); // 输出 "A",不是 "B"
- 静态方法调用由编译时类型决定,与对象实际类型无关
- 同理,访问静态字段也按声明类型解析,不存在“重写”概念
- 如果业务逻辑依赖运行时类型分发,不要用
static方法,改用实例方法 + 多态
static、非 private、非 final 的实例方法,且调用发生在运行时对象的实际类型上。最容易忽略的,其实是构造器阶段的方法调用时机,以及静态成员的绑定规则——这两处不看字节码几乎没法直观理解。










